@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,186 @@
|
|
|
1
|
+
import { inject } from '@angular/core';
|
|
2
|
+
import { CopilotChatConfigurationService } from '../core/chat-configuration/chat-configuration.service';
|
|
3
|
+
/**
|
|
4
|
+
* Watches chat configuration and provides reactive access to all configuration values.
|
|
5
|
+
* Must be called within an injection context.
|
|
6
|
+
*
|
|
7
|
+
* @returns Object with reactive signals and handler functions
|
|
8
|
+
*
|
|
9
|
+
* @example
|
|
10
|
+
* ```typescript
|
|
11
|
+
* export class ChatInputComponent {
|
|
12
|
+
* config = watchChatConfig();
|
|
13
|
+
*
|
|
14
|
+
* constructor() {
|
|
15
|
+
* effect(() => {
|
|
16
|
+
* const placeholder = this.config.labels().chatInputPlaceholder;
|
|
17
|
+
* console.log('Placeholder:', placeholder);
|
|
18
|
+
* });
|
|
19
|
+
* }
|
|
20
|
+
*
|
|
21
|
+
* handleSubmit(value: string) {
|
|
22
|
+
* this.config.submitInput(value);
|
|
23
|
+
* }
|
|
24
|
+
* }
|
|
25
|
+
* ```
|
|
26
|
+
*/
|
|
27
|
+
export function watchChatConfig() {
|
|
28
|
+
const service = inject(CopilotChatConfigurationService);
|
|
29
|
+
return {
|
|
30
|
+
labels: service.labels,
|
|
31
|
+
inputValue: service.inputValue,
|
|
32
|
+
submitInput: (value) => service.submitInput(value),
|
|
33
|
+
changeInput: (value) => service.changeInput(value)
|
|
34
|
+
};
|
|
35
|
+
}
|
|
36
|
+
/**
|
|
37
|
+
* Registers chat configuration within an injection context.
|
|
38
|
+
* Automatically updates the configuration when called.
|
|
39
|
+
*
|
|
40
|
+
* @param config - The configuration to register
|
|
41
|
+
*
|
|
42
|
+
* @example
|
|
43
|
+
* ```typescript
|
|
44
|
+
* export class ChatComponent {
|
|
45
|
+
* constructor() {
|
|
46
|
+
* registerChatConfig({
|
|
47
|
+
* labels: {
|
|
48
|
+
* chatInputPlaceholder: "How can I help?"
|
|
49
|
+
* },
|
|
50
|
+
* onSubmitInput: (value) => this.handleSubmit(value)
|
|
51
|
+
* });
|
|
52
|
+
* }
|
|
53
|
+
* }
|
|
54
|
+
* ```
|
|
55
|
+
*/
|
|
56
|
+
export function registerChatConfig(config) {
|
|
57
|
+
const service = inject(CopilotChatConfigurationService);
|
|
58
|
+
service.updateConfiguration(config);
|
|
59
|
+
}
|
|
60
|
+
/**
|
|
61
|
+
* Gets the current chat labels signal.
|
|
62
|
+
*
|
|
63
|
+
* @param service - The CopilotChatConfigurationService instance
|
|
64
|
+
* @returns Signal containing the current labels
|
|
65
|
+
*
|
|
66
|
+
* @example
|
|
67
|
+
* ```typescript
|
|
68
|
+
* export class ChatComponent {
|
|
69
|
+
* constructor(private chatConfig: CopilotChatConfigurationService) {
|
|
70
|
+
* const labels = getChatLabels(this.chatConfig);
|
|
71
|
+
* effect(() => {
|
|
72
|
+
* console.log('Current labels:', labels());
|
|
73
|
+
* });
|
|
74
|
+
* }
|
|
75
|
+
* }
|
|
76
|
+
* ```
|
|
77
|
+
*/
|
|
78
|
+
export function getChatLabels(service) {
|
|
79
|
+
return service.labels;
|
|
80
|
+
}
|
|
81
|
+
/**
|
|
82
|
+
* Updates chat labels.
|
|
83
|
+
*
|
|
84
|
+
* @param service - The CopilotChatConfigurationService instance
|
|
85
|
+
* @param labels - Partial labels to merge with defaults
|
|
86
|
+
*
|
|
87
|
+
* @example
|
|
88
|
+
* ```typescript
|
|
89
|
+
* export class ChatComponent {
|
|
90
|
+
* updatePlaceholder(text: string) {
|
|
91
|
+
* setChatLabels(this.chatConfig, {
|
|
92
|
+
* chatInputPlaceholder: text
|
|
93
|
+
* });
|
|
94
|
+
* }
|
|
95
|
+
* }
|
|
96
|
+
* ```
|
|
97
|
+
*/
|
|
98
|
+
export function setChatLabels(service, labels) {
|
|
99
|
+
service.setLabels(labels);
|
|
100
|
+
}
|
|
101
|
+
/**
|
|
102
|
+
* Gets the current input value signal.
|
|
103
|
+
*
|
|
104
|
+
* @param service - The CopilotChatConfigurationService instance
|
|
105
|
+
* @returns Signal containing the current input value
|
|
106
|
+
*
|
|
107
|
+
* @example
|
|
108
|
+
* ```typescript
|
|
109
|
+
* export class ChatInputComponent {
|
|
110
|
+
* inputValue = getChatInputValue(this.chatConfig);
|
|
111
|
+
*
|
|
112
|
+
* constructor(private chatConfig: CopilotChatConfigurationService) {
|
|
113
|
+
* effect(() => {
|
|
114
|
+
* const value = this.inputValue();
|
|
115
|
+
* if (value) {
|
|
116
|
+
* this.updateTextarea(value);
|
|
117
|
+
* }
|
|
118
|
+
* });
|
|
119
|
+
* }
|
|
120
|
+
* }
|
|
121
|
+
* ```
|
|
122
|
+
*/
|
|
123
|
+
export function getChatInputValue(service) {
|
|
124
|
+
return service.inputValue;
|
|
125
|
+
}
|
|
126
|
+
/**
|
|
127
|
+
* Sets the current input value.
|
|
128
|
+
*
|
|
129
|
+
* @param service - The CopilotChatConfigurationService instance
|
|
130
|
+
* @param value - The new input value
|
|
131
|
+
*
|
|
132
|
+
* @example
|
|
133
|
+
* ```typescript
|
|
134
|
+
* export class ChatInputComponent {
|
|
135
|
+
* onInputChange(event: Event) {
|
|
136
|
+
* const value = (event.target as HTMLInputElement).value;
|
|
137
|
+
* setChatInputValue(this.chatConfig, value);
|
|
138
|
+
* }
|
|
139
|
+
* }
|
|
140
|
+
* ```
|
|
141
|
+
*/
|
|
142
|
+
export function setChatInputValue(service, value) {
|
|
143
|
+
service.setInputValue(value);
|
|
144
|
+
}
|
|
145
|
+
/**
|
|
146
|
+
* Creates a chat configuration controller with dynamic update capabilities.
|
|
147
|
+
* This is useful when you need to programmatically manage configuration.
|
|
148
|
+
*
|
|
149
|
+
* @param service - The CopilotChatConfigurationService instance
|
|
150
|
+
* @param initialConfig - Optional initial configuration
|
|
151
|
+
* @returns Controller object with update and reset methods
|
|
152
|
+
*
|
|
153
|
+
* @example
|
|
154
|
+
* ```typescript
|
|
155
|
+
* export class ChatManagerComponent {
|
|
156
|
+
* chatController = createChatConfigController(this.chatConfig, {
|
|
157
|
+
* labels: { chatInputPlaceholder: "Ask me..." }
|
|
158
|
+
* });
|
|
159
|
+
*
|
|
160
|
+
* constructor(private chatConfig: CopilotChatConfigurationService) {}
|
|
161
|
+
*
|
|
162
|
+
* updateForSupportMode() {
|
|
163
|
+
* this.chatController.update({
|
|
164
|
+
* labels: { chatInputPlaceholder: "Describe your issue..." }
|
|
165
|
+
* });
|
|
166
|
+
* }
|
|
167
|
+
*
|
|
168
|
+
* resetToDefaults() {
|
|
169
|
+
* this.chatController.reset();
|
|
170
|
+
* }
|
|
171
|
+
* }
|
|
172
|
+
* ```
|
|
173
|
+
*/
|
|
174
|
+
export function createChatConfigController(service, initialConfig) {
|
|
175
|
+
// Apply initial configuration if provided
|
|
176
|
+
if (initialConfig) {
|
|
177
|
+
service.updateConfiguration(initialConfig);
|
|
178
|
+
}
|
|
179
|
+
return {
|
|
180
|
+
update: (config) => service.updateConfiguration(config),
|
|
181
|
+
reset: () => service.reset(),
|
|
182
|
+
getLabels: () => service.labels(),
|
|
183
|
+
getInputValue: () => service.inputValue()
|
|
184
|
+
};
|
|
185
|
+
}
|
|
186
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"chat-config.utils.js","sourceRoot":"","sources":["../../../src/utils/chat-config.utils.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,EAAU,MAAM,eAAe,CAAC;AAC/C,OAAO,EAAE,+BAA+B,EAAE,MAAM,uDAAuD,CAAC;AAMxG;;;;;;;;;;;;;;;;;;;;;;;GAuBG;AACH,MAAM,UAAU,eAAe;IAM7B,MAAM,OAAO,GAAG,MAAM,CAAC,+BAA+B,CAAC,CAAC;IAExD,OAAO;QACL,MAAM,EAAE,OAAO,CAAC,MAAM;QACtB,UAAU,EAAE,OAAO,CAAC,UAAU;QAC9B,WAAW,EAAE,CAAC,KAAa,EAAE,EAAE,CAAC,OAAO,CAAC,WAAW,CAAC,KAAK,CAAC;QAC1D,WAAW,EAAE,CAAC,KAAa,EAAE,EAAE,CAAC,OAAO,CAAC,WAAW,CAAC,KAAK,CAAC;KAC3D,CAAC;AACJ,CAAC;AAED;;;;;;;;;;;;;;;;;;;GAmBG;AACH,MAAM,UAAU,kBAAkB,CAAC,MAAgC;IACjE,MAAM,OAAO,GAAG,MAAM,CAAC,+BAA+B,CAAC,CAAC;IACxD,OAAO,CAAC,mBAAmB,CAAC,MAAM,CAAC,CAAC;AACtC,CAAC;AAED;;;;;;;;;;;;;;;;;GAiBG;AACH,MAAM,UAAU,aAAa,CAC3B,OAAwC;IAExC,OAAO,OAAO,CAAC,MAAM,CAAC;AACxB,CAAC;AAED;;;;;;;;;;;;;;;;GAgBG;AACH,MAAM,UAAU,aAAa,CAC3B,OAAwC,EACxC,MAAkC;IAElC,OAAO,CAAC,SAAS,CAAC,MAAM,CAAC,CAAC;AAC5B,CAAC;AAED;;;;;;;;;;;;;;;;;;;;;GAqBG;AACH,MAAM,UAAU,iBAAiB,CAC/B,OAAwC;IAExC,OAAO,OAAO,CAAC,UAAU,CAAC;AAC5B,CAAC;AAED;;;;;;;;;;;;;;;GAeG;AACH,MAAM,UAAU,iBAAiB,CAC/B,OAAwC,EACxC,KAAyB;IAEzB,OAAO,CAAC,aAAa,CAAC,KAAK,CAAC,CAAC;AAC/B,CAAC;AAED;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA4BG;AACH,MAAM,UAAU,0BAA0B,CACxC,OAAwC,EACxC,aAAwC;IAOxC,0CAA0C;IAC1C,IAAI,aAAa,EAAE,CAAC;QAClB,OAAO,CAAC,mBAAmB,CAAC,aAAa,CAAC,CAAC;IAC7C,CAAC;IAED,OAAO;QACL,MAAM,EAAE,CAAC,MAAgC,EAAE,EAAE,CAAC,OAAO,CAAC,mBAAmB,CAAC,MAAM,CAAC;QACjF,KAAK,EAAE,GAAG,EAAE,CAAC,OAAO,CAAC,KAAK,EAAE;QAC5B,SAAS,EAAE,GAAG,EAAE,CAAC,OAAO,CAAC,MAAM,EAAE;QACjC,aAAa,EAAE,GAAG,EAAE,CAAC,OAAO,CAAC,UAAU,EAAE;KAC1C,CAAC;AACJ,CAAC","sourcesContent":["import { inject, Signal } from '@angular/core';\nimport { CopilotChatConfigurationService } from '../core/chat-configuration/chat-configuration.service';\nimport { \n  CopilotChatConfiguration,\n  CopilotChatLabels\n} from '../core/chat-configuration/chat-configuration.types';\n\n/**\n * Watches chat configuration and provides reactive access to all configuration values.\n * Must be called within an injection context.\n * \n * @returns Object with reactive signals and handler functions\n * \n * @example\n * ```typescript\n * export class ChatInputComponent {\n *   config = watchChatConfig();\n *   \n *   constructor() {\n *     effect(() => {\n *       const placeholder = this.config.labels().chatInputPlaceholder;\n *       console.log('Placeholder:', placeholder);\n *     });\n *   }\n *   \n *   handleSubmit(value: string) {\n *     this.config.submitInput(value);\n *   }\n * }\n * ```\n */\nexport function watchChatConfig(): {\n  labels: Signal<CopilotChatLabels>;\n  inputValue: Signal<string | undefined>;\n  submitInput: (value: string) => void;\n  changeInput: (value: string) => void;\n} {\n  const service = inject(CopilotChatConfigurationService);\n  \n  return {\n    labels: service.labels,\n    inputValue: service.inputValue,\n    submitInput: (value: string) => service.submitInput(value),\n    changeInput: (value: string) => service.changeInput(value)\n  };\n}\n\n/**\n * Registers chat configuration within an injection context.\n * Automatically updates the configuration when called.\n * \n * @param config - The configuration to register\n * \n * @example\n * ```typescript\n * export class ChatComponent {\n *   constructor() {\n *     registerChatConfig({\n *       labels: {\n *         chatInputPlaceholder: \"How can I help?\"\n *       },\n *       onSubmitInput: (value) => this.handleSubmit(value)\n *     });\n *   }\n * }\n * ```\n */\nexport function registerChatConfig(config: CopilotChatConfiguration): void {\n  const service = inject(CopilotChatConfigurationService);\n  service.updateConfiguration(config);\n}\n\n/**\n * Gets the current chat labels signal.\n * \n * @param service - The CopilotChatConfigurationService instance\n * @returns Signal containing the current labels\n * \n * @example\n * ```typescript\n * export class ChatComponent {\n *   constructor(private chatConfig: CopilotChatConfigurationService) {\n *     const labels = getChatLabels(this.chatConfig);\n *     effect(() => {\n *       console.log('Current labels:', labels());\n *     });\n *   }\n * }\n * ```\n */\nexport function getChatLabels(\n  service: CopilotChatConfigurationService\n): Signal<CopilotChatLabels> {\n  return service.labels;\n}\n\n/**\n * Updates chat labels.\n * \n * @param service - The CopilotChatConfigurationService instance\n * @param labels - Partial labels to merge with defaults\n * \n * @example\n * ```typescript\n * export class ChatComponent {\n *   updatePlaceholder(text: string) {\n *     setChatLabels(this.chatConfig, {\n *       chatInputPlaceholder: text\n *     });\n *   }\n * }\n * ```\n */\nexport function setChatLabels(\n  service: CopilotChatConfigurationService,\n  labels: Partial<CopilotChatLabels>\n): void {\n  service.setLabels(labels);\n}\n\n/**\n * Gets the current input value signal.\n * \n * @param service - The CopilotChatConfigurationService instance\n * @returns Signal containing the current input value\n * \n * @example\n * ```typescript\n * export class ChatInputComponent {\n *   inputValue = getChatInputValue(this.chatConfig);\n *   \n *   constructor(private chatConfig: CopilotChatConfigurationService) {\n *     effect(() => {\n *       const value = this.inputValue();\n *       if (value) {\n *         this.updateTextarea(value);\n *       }\n *     });\n *   }\n * }\n * ```\n */\nexport function getChatInputValue(\n  service: CopilotChatConfigurationService\n): Signal<string | undefined> {\n  return service.inputValue;\n}\n\n/**\n * Sets the current input value.\n * \n * @param service - The CopilotChatConfigurationService instance\n * @param value - The new input value\n * \n * @example\n * ```typescript\n * export class ChatInputComponent {\n *   onInputChange(event: Event) {\n *     const value = (event.target as HTMLInputElement).value;\n *     setChatInputValue(this.chatConfig, value);\n *   }\n * }\n * ```\n */\nexport function setChatInputValue(\n  service: CopilotChatConfigurationService,\n  value: string | undefined\n): void {\n  service.setInputValue(value);\n}\n\n/**\n * Creates a chat configuration controller with dynamic update capabilities.\n * This is useful when you need to programmatically manage configuration.\n * \n * @param service - The CopilotChatConfigurationService instance\n * @param initialConfig - Optional initial configuration\n * @returns Controller object with update and reset methods\n * \n * @example\n * ```typescript\n * export class ChatManagerComponent {\n *   chatController = createChatConfigController(this.chatConfig, {\n *     labels: { chatInputPlaceholder: \"Ask me...\" }\n *   });\n *   \n *   constructor(private chatConfig: CopilotChatConfigurationService) {}\n *   \n *   updateForSupportMode() {\n *     this.chatController.update({\n *       labels: { chatInputPlaceholder: \"Describe your issue...\" }\n *     });\n *   }\n *   \n *   resetToDefaults() {\n *     this.chatController.reset();\n *   }\n * }\n * ```\n */\nexport function createChatConfigController(\n  service: CopilotChatConfigurationService,\n  initialConfig?: CopilotChatConfiguration\n): {\n  update: (config: CopilotChatConfiguration) => void;\n  reset: () => void;\n  getLabels: () => CopilotChatLabels;\n  getInputValue: () => string | undefined;\n} {\n  // Apply initial configuration if provided\n  if (initialConfig) {\n    service.updateConfiguration(initialConfig);\n  }\n  \n  return {\n    update: (config: CopilotChatConfiguration) => service.updateConfiguration(config),\n    reset: () => service.reset(),\n    getLabels: () => service.labels(),\n    getInputValue: () => service.inputValue()\n  };\n}"]}
|
|
@@ -0,0 +1,20 @@
|
|
|
1
|
+
import { inject } from "@angular/core";
|
|
2
|
+
import { CopilotKitService } from "../core/copilotkit.service";
|
|
3
|
+
/**
|
|
4
|
+
* Utility function to inject the CopilotKit service in a component or directive.
|
|
5
|
+
*
|
|
6
|
+
* @example
|
|
7
|
+
* ```typescript
|
|
8
|
+
* export class MyComponent {
|
|
9
|
+
* private copilotkit = injectCopilotKit();
|
|
10
|
+
*
|
|
11
|
+
* sendMessage() {
|
|
12
|
+
* this.copilotkit.copilotkit.sendMessage(...);
|
|
13
|
+
* }
|
|
14
|
+
* }
|
|
15
|
+
* ```
|
|
16
|
+
*/
|
|
17
|
+
export function injectCopilotKit() {
|
|
18
|
+
return inject(CopilotKitService);
|
|
19
|
+
}
|
|
20
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29waWxvdGtpdC51dGlscy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy91dGlscy9jb3BpbG90a2l0LnV0aWxzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxNQUFNLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDdkMsT0FBTyxFQUFFLGlCQUFpQixFQUFFLE1BQU0sNEJBQTRCLENBQUM7QUFFL0Q7Ozs7Ozs7Ozs7Ozs7R0FhRztBQUNILE1BQU0sVUFBVSxnQkFBZ0I7SUFDOUIsT0FBTyxNQUFNLENBQUMsaUJBQWlCLENBQUMsQ0FBQztBQUNuQyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgaW5qZWN0IH0gZnJvbSBcIkBhbmd1bGFyL2NvcmVcIjtcbmltcG9ydCB7IENvcGlsb3RLaXRTZXJ2aWNlIH0gZnJvbSBcIi4uL2NvcmUvY29waWxvdGtpdC5zZXJ2aWNlXCI7XG5cbi8qKlxuICogVXRpbGl0eSBmdW5jdGlvbiB0byBpbmplY3QgdGhlIENvcGlsb3RLaXQgc2VydmljZSBpbiBhIGNvbXBvbmVudCBvciBkaXJlY3RpdmUuXG4gKiBcbiAqIEBleGFtcGxlXG4gKiBgYGB0eXBlc2NyaXB0XG4gKiBleHBvcnQgY2xhc3MgTXlDb21wb25lbnQge1xuICogICBwcml2YXRlIGNvcGlsb3RraXQgPSBpbmplY3RDb3BpbG90S2l0KCk7XG4gKiAgIFxuICogICBzZW5kTWVzc2FnZSgpIHtcbiAqICAgICB0aGlzLmNvcGlsb3RraXQuY29waWxvdGtpdC5zZW5kTWVzc2FnZSguLi4pO1xuICogICB9XG4gKiB9XG4gKiBgYGBcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGluamVjdENvcGlsb3RLaXQoKSB7XG4gIHJldHVybiBpbmplY3QoQ29waWxvdEtpdFNlcnZpY2UpO1xufSJdfQ==
|
|
@@ -0,0 +1,228 @@
|
|
|
1
|
+
import { DestroyRef, inject } from '@angular/core';
|
|
2
|
+
import { CopilotKitService } from '../core/copilotkit.service';
|
|
3
|
+
import { z } from 'zod';
|
|
4
|
+
/**
|
|
5
|
+
* Explicitly adds a frontend tool to CopilotKit.
|
|
6
|
+
* Requires CopilotKitService to be passed as a parameter.
|
|
7
|
+
*
|
|
8
|
+
* @param service - The CopilotKitService instance
|
|
9
|
+
* @param tool - The tool to add
|
|
10
|
+
* @returns A cleanup function that removes the tool
|
|
11
|
+
*
|
|
12
|
+
* @example
|
|
13
|
+
* ```typescript
|
|
14
|
+
* export class MyComponent implements OnInit, OnDestroy {
|
|
15
|
+
* private cleanupFns: Array<() => void> = [];
|
|
16
|
+
*
|
|
17
|
+
* constructor(private copilotkit: CopilotKitService) {}
|
|
18
|
+
*
|
|
19
|
+
* ngOnInit() {
|
|
20
|
+
* const cleanup = addFrontendTool(this.copilotkit, {
|
|
21
|
+
* name: 'calculator',
|
|
22
|
+
* description: 'Performs calculations',
|
|
23
|
+
* parameters: z.object({
|
|
24
|
+
* expression: z.string()
|
|
25
|
+
* }),
|
|
26
|
+
* handler: async (args) => {
|
|
27
|
+
* return eval(args.expression);
|
|
28
|
+
* }
|
|
29
|
+
* });
|
|
30
|
+
*
|
|
31
|
+
* this.cleanupFns.push(cleanup);
|
|
32
|
+
* }
|
|
33
|
+
*
|
|
34
|
+
* ngOnDestroy() {
|
|
35
|
+
* this.cleanupFns.forEach(fn => fn());
|
|
36
|
+
* }
|
|
37
|
+
* }
|
|
38
|
+
* ```
|
|
39
|
+
*/
|
|
40
|
+
export function addFrontendTool(service, tool) {
|
|
41
|
+
// Add the tool to CopilotKit
|
|
42
|
+
service.copilotkit.addTool(tool);
|
|
43
|
+
// Register the render if provided
|
|
44
|
+
if (tool.render) {
|
|
45
|
+
const currentRenders = service.currentRenderToolCalls();
|
|
46
|
+
const existingIndex = currentRenders.findIndex((r) => r.name === tool.name);
|
|
47
|
+
if (existingIndex !== -1) {
|
|
48
|
+
console.error(`Tool with name '${tool.name}' already has a render. Skipping.`);
|
|
49
|
+
}
|
|
50
|
+
else {
|
|
51
|
+
const renderEntry = {
|
|
52
|
+
name: tool.name,
|
|
53
|
+
args: tool.parameters || z.object({}),
|
|
54
|
+
render: tool.render
|
|
55
|
+
};
|
|
56
|
+
service.setCurrentRenderToolCalls([...currentRenders, renderEntry]);
|
|
57
|
+
}
|
|
58
|
+
}
|
|
59
|
+
// Return cleanup function
|
|
60
|
+
return () => {
|
|
61
|
+
removeFrontendTool(service, tool.name);
|
|
62
|
+
};
|
|
63
|
+
}
|
|
64
|
+
/**
|
|
65
|
+
* Registers a frontend tool with CopilotKit and automatically removes it when the component/service is destroyed.
|
|
66
|
+
* Must be called within an injection context.
|
|
67
|
+
*
|
|
68
|
+
* @param tool - The tool to register
|
|
69
|
+
* @returns The tool name
|
|
70
|
+
*
|
|
71
|
+
* @example
|
|
72
|
+
* ```typescript
|
|
73
|
+
* export class MyComponent implements OnInit {
|
|
74
|
+
* ngOnInit() {
|
|
75
|
+
* // Automatically cleaned up on component destroy
|
|
76
|
+
* registerFrontendTool({
|
|
77
|
+
* name: 'search',
|
|
78
|
+
* description: 'Search for items',
|
|
79
|
+
* parameters: z.object({
|
|
80
|
+
* query: z.string()
|
|
81
|
+
* }),
|
|
82
|
+
* handler: async (args) => {
|
|
83
|
+
* return this.searchService.search(args.query);
|
|
84
|
+
* },
|
|
85
|
+
* render: SearchResultsComponent
|
|
86
|
+
* });
|
|
87
|
+
* }
|
|
88
|
+
* }
|
|
89
|
+
* ```
|
|
90
|
+
*/
|
|
91
|
+
export function registerFrontendTool(tool) {
|
|
92
|
+
const service = inject(CopilotKitService);
|
|
93
|
+
const destroyRef = inject(DestroyRef);
|
|
94
|
+
// Add the tool
|
|
95
|
+
service.copilotkit.addTool(tool);
|
|
96
|
+
// Register the render if provided
|
|
97
|
+
if (tool.render) {
|
|
98
|
+
const currentRenders = service.currentRenderToolCalls();
|
|
99
|
+
const existingIndex = currentRenders.findIndex((r) => r.name === tool.name);
|
|
100
|
+
if (existingIndex !== -1) {
|
|
101
|
+
console.error(`Tool with name '${tool.name}' already has a render. Skipping.`);
|
|
102
|
+
}
|
|
103
|
+
else {
|
|
104
|
+
const renderEntry = {
|
|
105
|
+
name: tool.name,
|
|
106
|
+
args: tool.parameters || z.object({}),
|
|
107
|
+
render: tool.render
|
|
108
|
+
};
|
|
109
|
+
service.setCurrentRenderToolCalls([...currentRenders, renderEntry]);
|
|
110
|
+
}
|
|
111
|
+
}
|
|
112
|
+
// Register cleanup with Angular's DestroyRef
|
|
113
|
+
destroyRef.onDestroy(() => {
|
|
114
|
+
removeFrontendTool(service, tool.name);
|
|
115
|
+
});
|
|
116
|
+
return tool.name;
|
|
117
|
+
}
|
|
118
|
+
/**
|
|
119
|
+
* Explicitly removes a frontend tool from CopilotKit.
|
|
120
|
+
*
|
|
121
|
+
* @param service - The CopilotKitService instance
|
|
122
|
+
* @param toolName - The name of the tool to remove
|
|
123
|
+
*
|
|
124
|
+
* @example
|
|
125
|
+
* ```typescript
|
|
126
|
+
* removeFrontendTool(this.copilotkit, 'calculator');
|
|
127
|
+
* ```
|
|
128
|
+
*/
|
|
129
|
+
export function removeFrontendTool(service, toolName) {
|
|
130
|
+
// Remove the tool
|
|
131
|
+
service.copilotkit.removeTool(toolName);
|
|
132
|
+
// Remove the render if it exists
|
|
133
|
+
const currentRenders = service.currentRenderToolCalls();
|
|
134
|
+
const filtered = currentRenders.filter((r) => r.name !== toolName);
|
|
135
|
+
if (filtered.length !== currentRenders.length) {
|
|
136
|
+
service.setCurrentRenderToolCalls(filtered);
|
|
137
|
+
}
|
|
138
|
+
}
|
|
139
|
+
/**
|
|
140
|
+
* Creates a frontend tool with dynamic parameters that can change over time.
|
|
141
|
+
* Uses Angular signals for reactivity.
|
|
142
|
+
*
|
|
143
|
+
* @param name - Tool name
|
|
144
|
+
* @param description - Tool description
|
|
145
|
+
* @param parameters - Zod schema for parameters
|
|
146
|
+
* @param handler - Signal or function that provides the handler
|
|
147
|
+
* @param render - Optional render component or template
|
|
148
|
+
* @returns Object with update and destroy methods
|
|
149
|
+
*
|
|
150
|
+
* @example
|
|
151
|
+
* ```typescript
|
|
152
|
+
* export class MyComponent {
|
|
153
|
+
* private toolConfig = signal({
|
|
154
|
+
* handler: async (args: any) => this.processDefault(args)
|
|
155
|
+
* });
|
|
156
|
+
*
|
|
157
|
+
* ngOnInit() {
|
|
158
|
+
* const tool = createDynamicFrontendTool(
|
|
159
|
+
* 'processor',
|
|
160
|
+
* 'Processes data',
|
|
161
|
+
* z.object({ data: z.string() }),
|
|
162
|
+
* () => this.toolConfig().handler
|
|
163
|
+
* );
|
|
164
|
+
*
|
|
165
|
+
* // Later, update the handler
|
|
166
|
+
* this.toolConfig.set({
|
|
167
|
+
* handler: async (args: any) => this.processAdvanced(args)
|
|
168
|
+
* });
|
|
169
|
+
* tool.update();
|
|
170
|
+
* }
|
|
171
|
+
* }
|
|
172
|
+
* ```
|
|
173
|
+
*/
|
|
174
|
+
export function createDynamicFrontendTool(name, description, parameters, handler, render) {
|
|
175
|
+
const service = inject(CopilotKitService);
|
|
176
|
+
const destroyRef = inject(DestroyRef);
|
|
177
|
+
let isRegistered = false;
|
|
178
|
+
const update = () => {
|
|
179
|
+
// Remove old tool if registered
|
|
180
|
+
if (isRegistered) {
|
|
181
|
+
service.copilotkit.removeTool(name);
|
|
182
|
+
}
|
|
183
|
+
// Create new tool configuration
|
|
184
|
+
const desc = typeof description === 'function' ? description() : description;
|
|
185
|
+
const currentHandler = handler();
|
|
186
|
+
const currentRender = render ? render() : undefined;
|
|
187
|
+
const tool = {
|
|
188
|
+
name,
|
|
189
|
+
description: desc,
|
|
190
|
+
parameters,
|
|
191
|
+
handler: currentHandler,
|
|
192
|
+
render: currentRender
|
|
193
|
+
};
|
|
194
|
+
// Add the tool
|
|
195
|
+
service.copilotkit.addTool(tool);
|
|
196
|
+
// Update render if provided
|
|
197
|
+
if (currentRender) {
|
|
198
|
+
const currentRenders = service.currentRenderToolCalls();
|
|
199
|
+
const renderEntry = {
|
|
200
|
+
name: name,
|
|
201
|
+
args: parameters,
|
|
202
|
+
render: currentRender
|
|
203
|
+
};
|
|
204
|
+
const existingIndex = currentRenders.findIndex((r) => r.name === name);
|
|
205
|
+
if (existingIndex !== -1) {
|
|
206
|
+
const updated = [...currentRenders];
|
|
207
|
+
updated[existingIndex] = renderEntry;
|
|
208
|
+
service.setCurrentRenderToolCalls(updated);
|
|
209
|
+
}
|
|
210
|
+
else {
|
|
211
|
+
service.setCurrentRenderToolCalls([...currentRenders, renderEntry]);
|
|
212
|
+
}
|
|
213
|
+
}
|
|
214
|
+
isRegistered = true;
|
|
215
|
+
};
|
|
216
|
+
const destroy = () => {
|
|
217
|
+
if (isRegistered) {
|
|
218
|
+
removeFrontendTool(service, name);
|
|
219
|
+
isRegistered = false;
|
|
220
|
+
}
|
|
221
|
+
};
|
|
222
|
+
// Initial setup
|
|
223
|
+
update();
|
|
224
|
+
// Register cleanup
|
|
225
|
+
destroyRef.onDestroy(destroy);
|
|
226
|
+
return { update, destroy };
|
|
227
|
+
}
|
|
228
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"frontend-tool.utils.js","sourceRoot":"","sources":["../../../src/utils/frontend-tool.utils.ts"],"names":[],"mappings":"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","sourcesContent":["import { DestroyRef, inject } from '@angular/core';\nimport { CopilotKitService } from '../core/copilotkit.service';\nimport { AngularFrontendTool, AngularToolCallRender, ToolCallRender } from '../core/copilotkit.types';\nimport { z } from 'zod';\n\n/**\n * Explicitly adds a frontend tool to CopilotKit.\n * Requires CopilotKitService to be passed as a parameter.\n * \n * @param service - The CopilotKitService instance\n * @param tool - The tool to add\n * @returns A cleanup function that removes the tool\n * \n * @example\n * ```typescript\n * export class MyComponent implements OnInit, OnDestroy {\n *   private cleanupFns: Array<() => void> = [];\n *   \n *   constructor(private copilotkit: CopilotKitService) {}\n *   \n *   ngOnInit() {\n *     const cleanup = addFrontendTool(this.copilotkit, {\n *       name: 'calculator',\n *       description: 'Performs calculations',\n *       parameters: z.object({\n *         expression: z.string()\n *       }),\n *       handler: async (args) => {\n *         return eval(args.expression);\n *       }\n *     });\n *     \n *     this.cleanupFns.push(cleanup);\n *   }\n *   \n *   ngOnDestroy() {\n *     this.cleanupFns.forEach(fn => fn());\n *   }\n * }\n * ```\n */\nexport function addFrontendTool<T extends Record<string, any> = Record<string, any>>(\n  service: CopilotKitService,\n  tool: AngularFrontendTool<T>\n): () => void {\n  // Add the tool to CopilotKit\n  service.copilotkit.addTool(tool);\n  \n  // Register the render if provided\n  if (tool.render) {\n    const currentRenders = service.currentRenderToolCalls();\n    const existingIndex = currentRenders.findIndex((r: ToolCallRender<unknown>) => r.name === tool.name);\n    \n    if (existingIndex !== -1) {\n      console.error(`Tool with name '${tool.name}' already has a render. Skipping.`);\n    } else {\n      const renderEntry: AngularToolCallRender<T> = {\n        name: tool.name,\n        args: tool.parameters || (z.object({}) as unknown as z.ZodSchema<T>),\n        render: tool.render\n      };\n      \n      service.setCurrentRenderToolCalls([...currentRenders, renderEntry]);\n    }\n  }\n  \n  // Return cleanup function\n  return () => {\n    removeFrontendTool(service, tool.name);\n  };\n}\n\n/**\n * Registers a frontend tool with CopilotKit and automatically removes it when the component/service is destroyed.\n * Must be called within an injection context.\n * \n * @param tool - The tool to register\n * @returns The tool name\n * \n * @example\n * ```typescript\n * export class MyComponent implements OnInit {\n *   ngOnInit() {\n *     // Automatically cleaned up on component destroy\n *     registerFrontendTool({\n *       name: 'search',\n *       description: 'Search for items',\n *       parameters: z.object({\n *         query: z.string()\n *       }),\n *       handler: async (args) => {\n *         return this.searchService.search(args.query);\n *       },\n *       render: SearchResultsComponent\n *     });\n *   }\n * }\n * ```\n */\nexport function registerFrontendTool<T extends Record<string, any> = Record<string, any>>(\n  tool: AngularFrontendTool<T>\n): string {\n  const service = inject(CopilotKitService);\n  const destroyRef = inject(DestroyRef);\n  \n  // Add the tool\n  service.copilotkit.addTool(tool);\n  \n  // Register the render if provided\n  if (tool.render) {\n    const currentRenders = service.currentRenderToolCalls();\n    const existingIndex = currentRenders.findIndex((r: ToolCallRender<unknown>) => r.name === tool.name);\n    \n    if (existingIndex !== -1) {\n      console.error(`Tool with name '${tool.name}' already has a render. Skipping.`);\n    } else {\n      const renderEntry: AngularToolCallRender<T> = {\n        name: tool.name,\n        args: tool.parameters || (z.object({}) as unknown as z.ZodSchema<T>),\n        render: tool.render\n      };\n      \n      service.setCurrentRenderToolCalls([...currentRenders, renderEntry]);\n    }\n  }\n  \n  // Register cleanup with Angular's DestroyRef\n  destroyRef.onDestroy(() => {\n    removeFrontendTool(service, tool.name);\n  });\n  \n  return tool.name;\n}\n\n/**\n * Explicitly removes a frontend tool from CopilotKit.\n * \n * @param service - The CopilotKitService instance\n * @param toolName - The name of the tool to remove\n * \n * @example\n * ```typescript\n * removeFrontendTool(this.copilotkit, 'calculator');\n * ```\n */\nexport function removeFrontendTool(\n  service: CopilotKitService,\n  toolName: string\n): void {\n  // Remove the tool\n  service.copilotkit.removeTool(toolName);\n  \n  // Remove the render if it exists\n  const currentRenders = service.currentRenderToolCalls();\n  const filtered = currentRenders.filter((r: ToolCallRender<unknown>) => r.name !== toolName);\n  if (filtered.length !== currentRenders.length) {\n    service.setCurrentRenderToolCalls(filtered);\n  }\n}\n\n/**\n * Creates a frontend tool with dynamic parameters that can change over time.\n * Uses Angular signals for reactivity.\n * \n * @param name - Tool name\n * @param description - Tool description\n * @param parameters - Zod schema for parameters\n * @param handler - Signal or function that provides the handler\n * @param render - Optional render component or template\n * @returns Object with update and destroy methods\n * \n * @example\n * ```typescript\n * export class MyComponent {\n *   private toolConfig = signal({\n *     handler: async (args: any) => this.processDefault(args)\n *   });\n *   \n *   ngOnInit() {\n *     const tool = createDynamicFrontendTool(\n *       'processor',\n *       'Processes data',\n *       z.object({ data: z.string() }),\n *       () => this.toolConfig().handler\n *     );\n *     \n *     // Later, update the handler\n *     this.toolConfig.set({\n *       handler: async (args: any) => this.processAdvanced(args)\n *     });\n *     tool.update();\n *   }\n * }\n * ```\n */\nexport function createDynamicFrontendTool<T extends Record<string, any> = Record<string, any>>(\n  name: string,\n  description: string | (() => string),\n  parameters: z.ZodSchema<T>,\n  handler: () => ((args: T) => Promise<any>),\n  render?: () => any\n): { update: () => void; destroy: () => void } {\n  const service = inject(CopilotKitService);\n  const destroyRef = inject(DestroyRef);\n  \n  let isRegistered = false;\n  \n  const update = () => {\n    // Remove old tool if registered\n    if (isRegistered) {\n      service.copilotkit.removeTool(name);\n    }\n    \n    // Create new tool configuration\n    const desc = typeof description === 'function' ? description() : description;\n    const currentHandler = handler();\n    const currentRender = render ? render() : undefined;\n    \n    const tool: AngularFrontendTool<T> = {\n      name,\n      description: desc,\n      parameters,\n      handler: currentHandler,\n      render: currentRender\n    };\n    \n    // Add the tool\n    service.copilotkit.addTool(tool);\n    \n    // Update render if provided\n    if (currentRender) {\n      const currentRenders = service.currentRenderToolCalls();\n      const renderEntry: AngularToolCallRender<T> = {\n        name: name,\n        args: parameters,\n        render: currentRender\n      };\n      \n      const existingIndex = currentRenders.findIndex((r: ToolCallRender<unknown>) => r.name === name);\n      if (existingIndex !== -1) {\n        const updated = [...currentRenders];\n        updated[existingIndex] = renderEntry;\n        service.setCurrentRenderToolCalls(updated);\n      } else {\n        service.setCurrentRenderToolCalls([...currentRenders, renderEntry]);\n      }\n    }\n    \n    isRegistered = true;\n  };\n  \n  const destroy = () => {\n    if (isRegistered) {\n      removeFrontendTool(service, name);\n      isRegistered = false;\n    }\n  };\n  \n  // Initial setup\n  update();\n  \n  // Register cleanup\n  destroyRef.onDestroy(destroy);\n  \n  return { update, destroy };\n}"]}
|