@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.
Files changed (171) hide show
  1. package/README.md +248 -0
  2. package/dist/README.md +248 -0
  3. package/dist/components/chat/copilot-chat-assistant-message.component.d.ts +10 -10
  4. package/dist/components/chat/copilot-chat-message-view.component.d.ts +42 -42
  5. package/dist/components/chat/copilot-chat-view.component.d.ts +14 -14
  6. package/dist/esm2022/components/chat/copilot-chat-assistant-message-buttons.component.mjs +384 -0
  7. package/dist/esm2022/components/chat/copilot-chat-assistant-message-renderer.component.mjs +286 -0
  8. package/dist/esm2022/components/chat/copilot-chat-assistant-message-toolbar.component.mjs +27 -0
  9. package/dist/esm2022/components/chat/copilot-chat-assistant-message.component.mjs +433 -0
  10. package/dist/esm2022/components/chat/copilot-chat-assistant-message.types.mjs +2 -0
  11. package/dist/esm2022/components/chat/copilot-chat-audio-recorder.component.mjs +202 -0
  12. package/dist/esm2022/components/chat/copilot-chat-buttons.component.mjs +321 -0
  13. package/dist/esm2022/components/chat/copilot-chat-input-defaults.mjs +38 -0
  14. package/dist/esm2022/components/chat/copilot-chat-input.component.mjs +666 -0
  15. package/dist/esm2022/components/chat/copilot-chat-input.types.mjs +10 -0
  16. package/dist/esm2022/components/chat/copilot-chat-message-view-cursor.component.mjs +45 -0
  17. package/dist/esm2022/components/chat/copilot-chat-message-view.component.mjs +296 -0
  18. package/dist/esm2022/components/chat/copilot-chat-message-view.types.mjs +2 -0
  19. package/dist/esm2022/components/chat/copilot-chat-textarea.component.mjs +188 -0
  20. package/dist/esm2022/components/chat/copilot-chat-tool-calls-view.component.mjs +216 -0
  21. package/dist/esm2022/components/chat/copilot-chat-toolbar.component.mjs +25 -0
  22. package/dist/esm2022/components/chat/copilot-chat-tools-menu.component.mjs +199 -0
  23. package/dist/esm2022/components/chat/copilot-chat-user-message-branch-navigation.component.mjs +137 -0
  24. package/dist/esm2022/components/chat/copilot-chat-user-message-buttons.component.mjs +207 -0
  25. package/dist/esm2022/components/chat/copilot-chat-user-message-renderer.component.mjs +35 -0
  26. package/dist/esm2022/components/chat/copilot-chat-user-message-toolbar.component.mjs +34 -0
  27. package/dist/esm2022/components/chat/copilot-chat-user-message.component.mjs +341 -0
  28. package/dist/esm2022/components/chat/copilot-chat-user-message.types.mjs +2 -0
  29. package/dist/esm2022/components/chat/copilot-chat-view-disclaimer.component.mjs +52 -0
  30. package/dist/esm2022/components/chat/copilot-chat-view-feather.component.mjs +55 -0
  31. package/dist/esm2022/components/chat/copilot-chat-view-handlers.service.mjs +19 -0
  32. package/dist/esm2022/components/chat/copilot-chat-view-input-container.component.mjs +110 -0
  33. package/dist/esm2022/components/chat/copilot-chat-view-scroll-to-bottom-button.component.mjs +93 -0
  34. package/dist/esm2022/components/chat/copilot-chat-view-scroll-view.component.mjs +443 -0
  35. package/dist/esm2022/components/chat/copilot-chat-view.component.mjs +479 -0
  36. package/dist/esm2022/components/chat/copilot-chat-view.types.mjs +2 -0
  37. package/dist/esm2022/components/chat/copilot-chat.component.mjs +214 -0
  38. package/dist/esm2022/components/copilotkit-tool-render.component.mjs +153 -0
  39. package/dist/esm2022/copilotkitnext-angular.mjs +5 -0
  40. package/dist/esm2022/core/chat-configuration/chat-configuration.providers.mjs +65 -0
  41. package/dist/esm2022/core/chat-configuration/chat-configuration.service.mjs +145 -0
  42. package/dist/esm2022/core/chat-configuration/chat-configuration.types.mjs +26 -0
  43. package/dist/esm2022/core/copilotkit.providers.mjs +34 -0
  44. package/dist/esm2022/core/copilotkit.service.mjs +430 -0
  45. package/dist/esm2022/core/copilotkit.types.mjs +12 -0
  46. package/dist/esm2022/directives/copilotkit-agent-context.directive.mjs +130 -0
  47. package/dist/esm2022/directives/copilotkit-agent.directive.mjs +217 -0
  48. package/dist/esm2022/directives/copilotkit-chat-config.directive.mjs +218 -0
  49. package/dist/esm2022/directives/copilotkit-config.directive.mjs +94 -0
  50. package/dist/esm2022/directives/copilotkit-frontend-tool.directive.mjs +130 -0
  51. package/dist/esm2022/directives/copilotkit-human-in-the-loop.directive.mjs +266 -0
  52. package/dist/esm2022/directives/stick-to-bottom.directive.mjs +181 -0
  53. package/dist/esm2022/index.mjs +70 -0
  54. package/dist/esm2022/lib/directives/tooltip.directive.mjs +211 -0
  55. package/dist/esm2022/lib/slots/copilot-slot.component.mjs +144 -0
  56. package/dist/esm2022/lib/slots/slot.types.mjs +6 -0
  57. package/dist/esm2022/lib/slots/slot.utils.mjs +222 -0
  58. package/dist/esm2022/lib/utils.mjs +10 -0
  59. package/dist/esm2022/services/resize-observer.service.mjs +152 -0
  60. package/dist/esm2022/services/scroll-position.service.mjs +124 -0
  61. package/dist/esm2022/types/frontend-tool.mjs +2 -0
  62. package/dist/esm2022/types/human-in-the-loop.mjs +2 -0
  63. package/dist/esm2022/utils/agent-context.utils.mjs +114 -0
  64. package/dist/esm2022/utils/agent.utils.mjs +204 -0
  65. package/dist/esm2022/utils/chat-config.utils.mjs +186 -0
  66. package/dist/esm2022/utils/copilotkit.utils.mjs +20 -0
  67. package/dist/esm2022/utils/frontend-tool.utils.mjs +228 -0
  68. package/dist/esm2022/utils/human-in-the-loop.utils.mjs +296 -0
  69. package/dist/fesm2022/copilotkitnext-angular.mjs +163 -164
  70. package/dist/fesm2022/copilotkitnext-angular.mjs.map +1 -1
  71. package/dist/styles.css +0 -27
  72. package/package.json +23 -20
  73. package/vitest.config.mts +32 -21
  74. package/.turbo/turbo-build.log +0 -39
  75. package/.turbo/turbo-check-types.log +0 -0
  76. package/.turbo/turbo-test.log +0 -71
  77. package/README-agent-context.md +0 -310
  78. package/ng-package.json +0 -19
  79. package/slots.md +0 -331
  80. package/src/components/chat/__tests__/copilot-chat-assistant-message.component.spec.ts +0 -282
  81. package/src/components/chat/__tests__/copilot-chat-input.component.spec.ts +0 -419
  82. package/src/components/chat/__tests__/copilot-chat-message-view.component.spec.ts +0 -372
  83. package/src/components/chat/__tests__/copilot-chat-user-message.component.spec.ts +0 -249
  84. package/src/components/chat/copilot-chat-assistant-message-buttons.component.ts +0 -292
  85. package/src/components/chat/copilot-chat-assistant-message-renderer.component.ts +0 -472
  86. package/src/components/chat/copilot-chat-assistant-message-toolbar.component.ts +0 -29
  87. package/src/components/chat/copilot-chat-assistant-message.component.ts +0 -463
  88. package/src/components/chat/copilot-chat-assistant-message.types.ts +0 -50
  89. package/src/components/chat/copilot-chat-audio-recorder.component.ts +0 -241
  90. package/src/components/chat/copilot-chat-buttons.component.ts +0 -308
  91. package/src/components/chat/copilot-chat-buttons.component.ts.bak +0 -471
  92. package/src/components/chat/copilot-chat-input-defaults.ts +0 -47
  93. package/src/components/chat/copilot-chat-input.component.ts +0 -512
  94. package/src/components/chat/copilot-chat-input.types.ts +0 -148
  95. package/src/components/chat/copilot-chat-message-view-cursor.component.ts +0 -51
  96. package/src/components/chat/copilot-chat-message-view.component.ts +0 -233
  97. package/src/components/chat/copilot-chat-message-view.types.ts +0 -39
  98. package/src/components/chat/copilot-chat-textarea.component.ts +0 -220
  99. package/src/components/chat/copilot-chat-tool-calls-view.component.ts +0 -261
  100. package/src/components/chat/copilot-chat-toolbar.component.ts +0 -35
  101. package/src/components/chat/copilot-chat-tools-menu.component.ts +0 -185
  102. package/src/components/chat/copilot-chat-user-message-branch-navigation.component.ts +0 -121
  103. package/src/components/chat/copilot-chat-user-message-buttons.component.ts +0 -170
  104. package/src/components/chat/copilot-chat-user-message-renderer.component.ts +0 -37
  105. package/src/components/chat/copilot-chat-user-message-toolbar.component.ts +0 -37
  106. package/src/components/chat/copilot-chat-user-message.component.ts +0 -247
  107. package/src/components/chat/copilot-chat-user-message.types.ts +0 -42
  108. package/src/components/chat/copilot-chat-view-disclaimer.component.ts +0 -51
  109. package/src/components/chat/copilot-chat-view-feather.component.ts +0 -47
  110. package/src/components/chat/copilot-chat-view-handlers.service.ts +0 -14
  111. package/src/components/chat/copilot-chat-view-input-container.component.ts +0 -87
  112. package/src/components/chat/copilot-chat-view-scroll-to-bottom-button.component.ts +0 -79
  113. package/src/components/chat/copilot-chat-view-scroll-view.component.ts +0 -322
  114. package/src/components/chat/copilot-chat-view.component.ts +0 -420
  115. package/src/components/chat/copilot-chat-view.types.ts +0 -52
  116. package/src/components/chat/copilot-chat.component.ts +0 -232
  117. package/src/components/copilotkit-tool-render.component.ts +0 -169
  118. package/src/core/__tests__/copilotkit.service.spec.ts +0 -1051
  119. package/src/core/__tests__/copilotkit.service.wildcard.spec.ts +0 -316
  120. package/src/core/chat-configuration/__tests__/chat-configuration.service.spec.ts +0 -287
  121. package/src/core/chat-configuration/chat-configuration.providers.ts +0 -71
  122. package/src/core/chat-configuration/chat-configuration.service.ts +0 -162
  123. package/src/core/chat-configuration/chat-configuration.types.ts +0 -57
  124. package/src/core/copilotkit.providers.ts +0 -59
  125. package/src/core/copilotkit.service.ts +0 -542
  126. package/src/core/copilotkit.types.ts +0 -132
  127. package/src/directives/__tests__/copilotkit-agent-context.directive.spec.ts +0 -384
  128. package/src/directives/__tests__/copilotkit-agent.directive.spec.ts +0 -253
  129. package/src/directives/__tests__/copilotkit-chat-config.directive.spec.ts +0 -385
  130. package/src/directives/__tests__/copilotkit-config.directive.spec.ts +0 -69
  131. package/src/directives/__tests__/copilotkit-frontend-tool-simple.directive.spec.ts +0 -60
  132. package/src/directives/__tests__/copilotkit-frontend-tool.directive.spec.ts +0 -108
  133. package/src/directives/__tests__/copilotkit-human-in-the-loop.directive.spec.ts +0 -452
  134. package/src/directives/copilotkit-agent-context.directive.ts +0 -138
  135. package/src/directives/copilotkit-agent.directive.ts +0 -225
  136. package/src/directives/copilotkit-chat-config.directive.ts +0 -241
  137. package/src/directives/copilotkit-config.directive.ts +0 -81
  138. package/src/directives/copilotkit-frontend-tool.directive.ts +0 -145
  139. package/src/directives/copilotkit-human-in-the-loop.directive.ts +0 -281
  140. package/src/directives/stick-to-bottom.directive.ts +0 -204
  141. package/src/index.ts +0 -105
  142. package/src/lib/directives/tooltip.directive.ts +0 -292
  143. package/src/lib/slots/__tests__/slot.utils.spec.ts +0 -377
  144. package/src/lib/slots/copilot-slot.component.ts +0 -135
  145. package/src/lib/slots/index.ts +0 -3
  146. package/src/lib/slots/slot.types.ts +0 -64
  147. package/src/lib/slots/slot.utils.ts +0 -289
  148. package/src/lib/utils.ts +0 -10
  149. package/src/public-api.ts +0 -1
  150. package/src/services/resize-observer.service.ts +0 -181
  151. package/src/services/scroll-position.service.ts +0 -169
  152. package/src/styles/globals.css +0 -266
  153. package/src/styles/index.css +0 -3
  154. package/src/test-setup.ts +0 -15
  155. package/src/testing/index.ts +0 -3
  156. package/src/testing/testing.utils.ts +0 -248
  157. package/src/types/frontend-tool.ts +0 -44
  158. package/src/types/human-in-the-loop.ts +0 -52
  159. package/src/utils/__tests__/agent.utils.spec.ts +0 -234
  160. package/src/utils/__tests__/chat-config.utils.spec.ts +0 -306
  161. package/src/utils/__tests__/frontend-tool-inject.spec.ts +0 -350
  162. package/src/utils/__tests__/frontend-tool-integration.spec.ts +0 -199
  163. package/src/utils/__tests__/frontend-tool.utils.spec.ts +0 -272
  164. package/src/utils/__tests__/human-in-the-loop.utils.spec.ts +0 -365
  165. package/src/utils/agent-context.utils.ts +0 -133
  166. package/src/utils/agent.utils.ts +0 -239
  167. package/src/utils/chat-config.utils.ts +0 -221
  168. package/src/utils/copilotkit.utils.ts +0 -20
  169. package/src/utils/frontend-tool.utils.ts +0 -266
  170. package/src/utils/human-in-the-loop.utils.ts +0 -359
  171. package/tsconfig.spec.json +0 -12
@@ -0,0 +1,296 @@
1
+ import { DestroyRef, inject, signal } from '@angular/core';
2
+ import { CopilotKitService } from '../core/copilotkit.service';
3
+ import { ToolCallStatus } from '../core/copilotkit.types';
4
+ /**
5
+ * Registers a human-in-the-loop tool that requires user interaction.
6
+ * Must be called within an injection context.
7
+ * Automatically cleans up when the component/service is destroyed.
8
+ *
9
+ * @param tool - The human-in-the-loop tool configuration
10
+ * @returns The tool ID
11
+ *
12
+ * @example
13
+ * ```typescript
14
+ * export class ApprovalComponent {
15
+ * toolId = registerHumanInTheLoop({
16
+ * name: 'requireApproval',
17
+ * description: 'Requires user approval',
18
+ * args: z.object({ action: z.string() }),
19
+ * render: ApprovalDialogComponent
20
+ * });
21
+ * }
22
+ * ```
23
+ */
24
+ export function registerHumanInTheLoop(tool) {
25
+ const service = inject(CopilotKitService);
26
+ const destroyRef = inject(DestroyRef);
27
+ // Create state management
28
+ const statusSignal = signal(ToolCallStatus.InProgress);
29
+ let resolvePromise = null;
30
+ // Create respond function
31
+ const respond = async (result) => {
32
+ if (resolvePromise) {
33
+ resolvePromise(result);
34
+ statusSignal.set(ToolCallStatus.Complete);
35
+ resolvePromise = null;
36
+ }
37
+ };
38
+ // Create handler that returns a Promise
39
+ const handler = async (_args) => {
40
+ return new Promise((resolve) => {
41
+ statusSignal.set(ToolCallStatus.Executing);
42
+ resolvePromise = resolve;
43
+ });
44
+ };
45
+ // Create enhanced render function
46
+ const enhancedRender = createEnhancedRender(tool.render, statusSignal, respond);
47
+ // Create the frontend tool
48
+ const frontendTool = {
49
+ ...tool,
50
+ handler,
51
+ render: enhancedRender
52
+ };
53
+ // Add the tool (returns void, so we use the tool name as ID)
54
+ service.copilotkit.addTool(frontendTool);
55
+ const toolId = frontendTool.name;
56
+ // Register tool render if provided
57
+ if (frontendTool.render && tool.parameters) {
58
+ service.registerToolRender(frontendTool.name, {
59
+ name: frontendTool.name,
60
+ args: tool.parameters,
61
+ render: frontendTool.render
62
+ });
63
+ }
64
+ // Cleanup on destroy
65
+ destroyRef.onDestroy(() => {
66
+ service.copilotkit.removeTool(toolId);
67
+ if (frontendTool.render) {
68
+ service.unregisterToolRender(frontendTool.name);
69
+ }
70
+ });
71
+ return toolId;
72
+ }
73
+ /**
74
+ * Adds a human-in-the-loop tool with explicit service parameter.
75
+ * Returns a cleanup function.
76
+ *
77
+ * @param service - The CopilotKitService instance
78
+ * @param tool - The human-in-the-loop tool configuration
79
+ * @returns Cleanup function to remove the tool
80
+ *
81
+ * @example
82
+ * ```typescript
83
+ * export class MyComponent implements OnInit, OnDestroy {
84
+ * private cleanup?: () => void;
85
+ *
86
+ * constructor(private copilotkit: CopilotKitService) {}
87
+ *
88
+ * ngOnInit() {
89
+ * this.cleanup = addHumanInTheLoop(this.copilotkit, {
90
+ * name: 'requireApproval',
91
+ * description: 'Requires user approval',
92
+ * args: z.object({ action: z.string() }),
93
+ * render: ApprovalDialogComponent
94
+ * });
95
+ * }
96
+ *
97
+ * ngOnDestroy() {
98
+ * this.cleanup?.();
99
+ * }
100
+ * }
101
+ * ```
102
+ */
103
+ export function addHumanInTheLoop(service, tool) {
104
+ // Create state management
105
+ const statusSignal = signal(ToolCallStatus.InProgress);
106
+ let resolvePromise = null;
107
+ // Create respond function
108
+ const respond = async (result) => {
109
+ if (resolvePromise) {
110
+ resolvePromise(result);
111
+ statusSignal.set(ToolCallStatus.Complete);
112
+ resolvePromise = null;
113
+ }
114
+ };
115
+ // Create handler that returns a Promise
116
+ const handler = async (_args) => {
117
+ return new Promise((resolve) => {
118
+ statusSignal.set(ToolCallStatus.Executing);
119
+ resolvePromise = resolve;
120
+ });
121
+ };
122
+ // Create enhanced render function
123
+ const enhancedRender = createEnhancedRender(tool.render, statusSignal, respond);
124
+ // Create the frontend tool
125
+ const frontendTool = {
126
+ ...tool,
127
+ handler,
128
+ render: enhancedRender
129
+ };
130
+ // Add the tool (returns void, so we use the tool name as ID)
131
+ service.copilotkit.addTool(frontendTool);
132
+ const toolId = frontendTool.name;
133
+ // Register tool render if provided
134
+ if (frontendTool.render && tool.parameters) {
135
+ service.registerToolRender(frontendTool.name, {
136
+ name: frontendTool.name,
137
+ args: tool.parameters,
138
+ render: frontendTool.render
139
+ });
140
+ }
141
+ // Return cleanup function
142
+ return () => {
143
+ service.copilotkit.removeTool(toolId);
144
+ if (frontendTool.render) {
145
+ service.unregisterToolRender(frontendTool.name);
146
+ }
147
+ };
148
+ }
149
+ /**
150
+ * Creates a human-in-the-loop tool with dynamic update capabilities.
151
+ *
152
+ * @param service - The CopilotKitService instance
153
+ * @param tool - The human-in-the-loop tool configuration
154
+ * @returns Object with status signal, update and destroy methods
155
+ *
156
+ * @example
157
+ * ```typescript
158
+ * export class MyComponent {
159
+ * humanInTheLoop = createHumanInTheLoop(this.copilotkit, {
160
+ * name: 'requireApproval',
161
+ * description: 'Requires user approval',
162
+ * args: z.object({ action: z.string() }),
163
+ * render: ApprovalDialogComponent
164
+ * });
165
+ *
166
+ * updateDescription(newDesc: string) {
167
+ * this.humanInTheLoop.update({ description: newDesc });
168
+ * }
169
+ *
170
+ * ngOnDestroy() {
171
+ * this.humanInTheLoop.destroy();
172
+ * }
173
+ * }
174
+ * ```
175
+ */
176
+ export function createHumanInTheLoop(service, tool) {
177
+ // Create state management
178
+ const statusSignal = signal(ToolCallStatus.InProgress);
179
+ let currentTool = { ...tool };
180
+ let toolId = '';
181
+ let resolvePromise = null;
182
+ // Create respond function
183
+ const respond = async (result) => {
184
+ if (resolvePromise) {
185
+ resolvePromise(result);
186
+ statusSignal.set(ToolCallStatus.Complete);
187
+ resolvePromise = null;
188
+ }
189
+ };
190
+ // Create handler that returns a Promise
191
+ const handler = async (_args) => {
192
+ return new Promise((resolve) => {
193
+ statusSignal.set(ToolCallStatus.Executing);
194
+ resolvePromise = resolve;
195
+ });
196
+ };
197
+ // Function to add the tool
198
+ const addTool = () => {
199
+ // Create enhanced render function
200
+ const enhancedRender = createEnhancedRender(currentTool.render, statusSignal, respond);
201
+ // Create the frontend tool
202
+ const frontendTool = {
203
+ ...currentTool,
204
+ handler,
205
+ render: enhancedRender
206
+ };
207
+ // Add tool (returns void, so we use the tool name as ID)
208
+ service.copilotkit.addTool(frontendTool);
209
+ toolId = frontendTool.name;
210
+ // Register tool render if provided
211
+ if (frontendTool.render && currentTool.parameters) {
212
+ service.registerToolRender(frontendTool.name, {
213
+ name: frontendTool.name,
214
+ args: currentTool.parameters,
215
+ render: frontendTool.render
216
+ });
217
+ }
218
+ };
219
+ // Initialize the tool
220
+ addTool();
221
+ return {
222
+ status: statusSignal.asReadonly(),
223
+ toolId,
224
+ update: (updates) => {
225
+ // Remove old tool
226
+ service.copilotkit.removeTool(toolId);
227
+ if (currentTool.render) {
228
+ service.unregisterToolRender(currentTool.name);
229
+ }
230
+ // Update tool configuration
231
+ currentTool = { ...currentTool, ...updates };
232
+ // Re-add with new configuration
233
+ addTool();
234
+ },
235
+ destroy: () => {
236
+ service.copilotkit.removeTool(toolId);
237
+ if (currentTool.render) {
238
+ service.unregisterToolRender(currentTool.name);
239
+ }
240
+ }
241
+ };
242
+ }
243
+ /**
244
+ * Creates an enhanced render function that injects the respond function
245
+ * when the status is 'executing'.
246
+ */
247
+ function createEnhancedRender(originalRender, _statusSignal, _respond) {
248
+ // For component classes, we need to create a wrapper
249
+ if (isComponentClass(originalRender)) {
250
+ // Return a wrapper component factory
251
+ // This is complex in Angular and would require dynamic component creation
252
+ // For now, we'll return the original and rely on prop injection
253
+ return originalRender;
254
+ }
255
+ // For templates, we can't easily wrap them
256
+ // The template context will be enhanced in the render component
257
+ return originalRender;
258
+ }
259
+ /**
260
+ * Helper function to check if a value is a component class
261
+ */
262
+ function isComponentClass(value) {
263
+ return typeof value === 'function' && value.prototype;
264
+ }
265
+ /**
266
+ * Enhanced component wrapper for human-in-the-loop.
267
+ * This would be used internally by the tool render component to inject
268
+ * the respond function based on status.
269
+ *
270
+ * @internal
271
+ */
272
+ export function enhancePropsForHumanInTheLoop(props, status, respond) {
273
+ if (status === ToolCallStatus.Executing && respond) {
274
+ return {
275
+ ...props,
276
+ status: ToolCallStatus.Executing,
277
+ respond
278
+ };
279
+ }
280
+ if (status === ToolCallStatus.Complete) {
281
+ return {
282
+ ...props,
283
+ status: ToolCallStatus.Complete,
284
+ result: typeof props.result === 'string' ? props.result : '',
285
+ respond: undefined
286
+ };
287
+ }
288
+ // InProgress
289
+ return {
290
+ ...props,
291
+ status: ToolCallStatus.InProgress,
292
+ result: undefined,
293
+ respond: undefined
294
+ };
295
+ }
296
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"human-in-the-loop.utils.js","sourceRoot":"","sources":["../../../src/utils/human-in-the-loop.utils.ts"],"names":[],"mappings":"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","sourcesContent":["import { \n  DestroyRef, \n  inject, \n  signal, \n  Signal,\n  Type,\n  TemplateRef\n} from '@angular/core';\nimport { CopilotKitService } from '../core/copilotkit.service';\nimport { \n  AngularHumanInTheLoop, \n  ToolCallStatus,\n  HumanInTheLoopState,\n  HumanInTheLoopProps,\n  AngularFrontendTool\n} from '../core/copilotkit.types';\n\n/**\n * Registers a human-in-the-loop tool that requires user interaction.\n * Must be called within an injection context.\n * Automatically cleans up when the component/service is destroyed.\n * \n * @param tool - The human-in-the-loop tool configuration\n * @returns The tool ID\n * \n * @example\n * ```typescript\n * export class ApprovalComponent {\n *   toolId = registerHumanInTheLoop({\n *     name: 'requireApproval',\n *     description: 'Requires user approval',\n *     args: z.object({ action: z.string() }),\n *     render: ApprovalDialogComponent\n *   });\n * }\n * ```\n */\nexport function registerHumanInTheLoop<T extends Record<string, any> = Record<string, any>>(\n  tool: AngularHumanInTheLoop<T>\n): string {\n  const service = inject(CopilotKitService);\n  const destroyRef = inject(DestroyRef);\n  \n  // Create state management\n  const statusSignal = signal<ToolCallStatus>(ToolCallStatus.InProgress);\n  let resolvePromise: ((result: unknown) => void) | null = null;\n  \n  // Create respond function\n  const respond = async (result: unknown): Promise<void> => {\n    if (resolvePromise) {\n      resolvePromise(result);\n      statusSignal.set(ToolCallStatus.Complete);\n      resolvePromise = null;\n    }\n  };\n  \n  // Create handler that returns a Promise\n  const handler = async (_args: T): Promise<unknown> => {\n    return new Promise((resolve) => {\n      statusSignal.set(ToolCallStatus.Executing);\n      resolvePromise = resolve;\n    });\n  };\n  \n  // Create enhanced render function\n  const enhancedRender = createEnhancedRender(tool.render, statusSignal, respond);\n  \n  // Create the frontend tool\n  const frontendTool: AngularFrontendTool<T> = {\n    ...tool,\n    handler,\n    render: enhancedRender\n  };\n  \n  // Add the tool (returns void, so we use the tool name as ID)\n  service.copilotkit.addTool(frontendTool);\n  const toolId = frontendTool.name;\n  \n  // Register tool render if provided\n  if (frontendTool.render && tool.parameters) {\n    service.registerToolRender(frontendTool.name, {\n      name: frontendTool.name,\n      args: tool.parameters,\n      render: frontendTool.render\n    });\n  }\n  \n  // Cleanup on destroy\n  destroyRef.onDestroy(() => {\n    service.copilotkit.removeTool(toolId);\n    if (frontendTool.render) {\n      service.unregisterToolRender(frontendTool.name);\n    }\n  });\n  \n  return toolId;\n}\n\n/**\n * Adds a human-in-the-loop tool with explicit service parameter.\n * Returns a cleanup function.\n * \n * @param service - The CopilotKitService instance\n * @param tool - The human-in-the-loop tool configuration\n * @returns Cleanup function to remove the tool\n * \n * @example\n * ```typescript\n * export class MyComponent implements OnInit, OnDestroy {\n *   private cleanup?: () => void;\n *   \n *   constructor(private copilotkit: CopilotKitService) {}\n *   \n *   ngOnInit() {\n *     this.cleanup = addHumanInTheLoop(this.copilotkit, {\n *       name: 'requireApproval',\n *       description: 'Requires user approval',\n *       args: z.object({ action: z.string() }),\n *       render: ApprovalDialogComponent\n *     });\n *   }\n *   \n *   ngOnDestroy() {\n *     this.cleanup?.();\n *   }\n * }\n * ```\n */\nexport function addHumanInTheLoop<T extends Record<string, any> = Record<string, any>>(\n  service: CopilotKitService,\n  tool: AngularHumanInTheLoop<T>\n): () => void {\n  // Create state management\n  const statusSignal = signal<ToolCallStatus>(ToolCallStatus.InProgress);\n  let resolvePromise: ((result: unknown) => void) | null = null;\n  \n  // Create respond function\n  const respond = async (result: unknown): Promise<void> => {\n    if (resolvePromise) {\n      resolvePromise(result);\n      statusSignal.set(ToolCallStatus.Complete);\n      resolvePromise = null;\n    }\n  };\n  \n  // Create handler that returns a Promise\n  const handler = async (_args: T): Promise<unknown> => {\n    return new Promise((resolve) => {\n      statusSignal.set(ToolCallStatus.Executing);\n      resolvePromise = resolve;\n    });\n  };\n  \n  // Create enhanced render function\n  const enhancedRender = createEnhancedRender(tool.render, statusSignal, respond);\n  \n  // Create the frontend tool\n  const frontendTool: AngularFrontendTool<T> = {\n    ...tool,\n    handler,\n    render: enhancedRender\n  };\n  \n  // Add the tool (returns void, so we use the tool name as ID)\n  service.copilotkit.addTool(frontendTool);\n  const toolId = frontendTool.name;\n  \n  // Register tool render if provided\n  if (frontendTool.render && tool.parameters) {\n    service.registerToolRender(frontendTool.name, {\n      name: frontendTool.name,\n      args: tool.parameters,\n      render: frontendTool.render\n    });\n  }\n  \n  // Return cleanup function\n  return () => {\n    service.copilotkit.removeTool(toolId);\n    if (frontendTool.render) {\n      service.unregisterToolRender(frontendTool.name);\n    }\n  };\n}\n\n/**\n * Creates a human-in-the-loop tool with dynamic update capabilities.\n * \n * @param service - The CopilotKitService instance\n * @param tool - The human-in-the-loop tool configuration\n * @returns Object with status signal, update and destroy methods\n * \n * @example\n * ```typescript\n * export class MyComponent {\n *   humanInTheLoop = createHumanInTheLoop(this.copilotkit, {\n *     name: 'requireApproval',\n *     description: 'Requires user approval',\n *     args: z.object({ action: z.string() }),\n *     render: ApprovalDialogComponent\n *   });\n *   \n *   updateDescription(newDesc: string) {\n *     this.humanInTheLoop.update({ description: newDesc });\n *   }\n *   \n *   ngOnDestroy() {\n *     this.humanInTheLoop.destroy();\n *   }\n * }\n * ```\n */\nexport function createHumanInTheLoop<T extends Record<string, any> = Record<string, any>>(\n  service: CopilotKitService,\n  tool: AngularHumanInTheLoop<T>\n): HumanInTheLoopState & { update: (updates: Partial<AngularHumanInTheLoop<T>>) => void } {\n  // Create state management\n  const statusSignal = signal<ToolCallStatus>(ToolCallStatus.InProgress);\n  let currentTool = { ...tool };\n  let toolId: string = '';\n  let resolvePromise: ((result: unknown) => void) | null = null;\n  \n  // Create respond function\n  const respond = async (result: unknown): Promise<void> => {\n    if (resolvePromise) {\n      resolvePromise(result);\n      statusSignal.set(ToolCallStatus.Complete);\n      resolvePromise = null;\n    }\n  };\n  \n  // Create handler that returns a Promise\n  const handler = async (_args: T): Promise<unknown> => {\n    return new Promise((resolve) => {\n      statusSignal.set(ToolCallStatus.Executing);\n      resolvePromise = resolve;\n    });\n  };\n  \n  // Function to add the tool\n  const addTool = () => {\n    // Create enhanced render function\n    const enhancedRender = createEnhancedRender(currentTool.render, statusSignal, respond);\n    \n    // Create the frontend tool\n    const frontendTool: AngularFrontendTool<T> = {\n      ...currentTool,\n      handler,\n      render: enhancedRender\n    };\n    \n    // Add tool (returns void, so we use the tool name as ID)\n    service.copilotkit.addTool(frontendTool);\n    toolId = frontendTool.name;\n    \n    // Register tool render if provided\n    if (frontendTool.render && currentTool.parameters) {\n      service.registerToolRender(frontendTool.name, {\n        name: frontendTool.name,\n        args: currentTool.parameters,\n        render: frontendTool.render\n      });\n    }\n  };\n  \n  // Initialize the tool\n  addTool();\n  \n  return {\n    status: statusSignal.asReadonly(),\n    toolId,\n    update: (updates: Partial<AngularHumanInTheLoop<T>>) => {\n      // Remove old tool\n      service.copilotkit.removeTool(toolId);\n      if (currentTool.render) {\n        service.unregisterToolRender(currentTool.name);\n      }\n      \n      // Update tool configuration\n      currentTool = { ...currentTool, ...updates };\n      \n      // Re-add with new configuration\n      addTool();\n    },\n    destroy: () => {\n      service.copilotkit.removeTool(toolId);\n      if (currentTool.render) {\n        service.unregisterToolRender(currentTool.name);\n      }\n    }\n  };\n}\n\n/**\n * Creates an enhanced render function that injects the respond function\n * when the status is 'executing'.\n */\nfunction createEnhancedRender<T extends Record<string, any>>(\n  originalRender: Type<any> | TemplateRef<HumanInTheLoopProps<T>>,\n  _statusSignal: Signal<ToolCallStatus>,\n  _respond: (result: unknown) => Promise<void>\n): Type<any> | TemplateRef<any> {\n  // For component classes, we need to create a wrapper\n  if (isComponentClass(originalRender)) {\n    // Return a wrapper component factory\n    // This is complex in Angular and would require dynamic component creation\n    // For now, we'll return the original and rely on prop injection\n    return originalRender;\n  }\n  \n  // For templates, we can't easily wrap them\n  // The template context will be enhanced in the render component\n  return originalRender;\n}\n\n/**\n * Helper function to check if a value is a component class\n */\nfunction isComponentClass(value: any): value is Type<any> {\n  return typeof value === 'function' && value.prototype;\n}\n\n/**\n * Enhanced component wrapper for human-in-the-loop.\n * This would be used internally by the tool render component to inject\n * the respond function based on status.\n * \n * @internal\n */\nexport function enhancePropsForHumanInTheLoop<T>(\n  props: HumanInTheLoopProps<T>,\n  status: ToolCallStatus,\n  respond?: (result: unknown) => Promise<void>\n): HumanInTheLoopProps<T> {\n  if (status === ToolCallStatus.Executing && respond) {\n    return {\n      ...props,\n      status: ToolCallStatus.Executing,\n      respond\n    } as HumanInTheLoopProps<T>;\n  }\n  \n  if (status === ToolCallStatus.Complete) {\n    return {\n      ...props,\n      status: ToolCallStatus.Complete,\n      result: typeof props.result === 'string' ? props.result : '',\n      respond: undefined\n    } as HumanInTheLoopProps<T>;\n  }\n  \n  // InProgress\n  return {\n    ...props,\n    status: ToolCallStatus.InProgress,\n    result: undefined,\n    respond: undefined\n  } as HumanInTheLoopProps<T>;\n}"]}