@copilotkitnext/angular 0.0.2 → 0.0.5

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (173) hide show
  1. package/README.md +3 -3
  2. package/dist/README.md +3 -3
  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/core/copilotkit.providers.d.ts +1 -1
  7. package/dist/core/copilotkit.service.d.ts +5 -5
  8. package/dist/core/copilotkit.types.d.ts +8 -10
  9. package/dist/directives/copilotkit-frontend-tool.directive.d.ts +1 -1
  10. package/dist/esm2022/components/chat/copilot-chat-assistant-message-buttons.component.mjs +384 -0
  11. package/dist/esm2022/components/chat/copilot-chat-assistant-message-renderer.component.mjs +286 -0
  12. package/dist/esm2022/components/chat/copilot-chat-assistant-message-toolbar.component.mjs +27 -0
  13. package/dist/esm2022/components/chat/copilot-chat-assistant-message.component.mjs +433 -0
  14. package/dist/esm2022/components/chat/copilot-chat-assistant-message.types.mjs +2 -0
  15. package/dist/esm2022/components/chat/copilot-chat-audio-recorder.component.mjs +202 -0
  16. package/dist/esm2022/components/chat/copilot-chat-buttons.component.mjs +321 -0
  17. package/dist/esm2022/components/chat/copilot-chat-input-defaults.mjs +38 -0
  18. package/dist/esm2022/components/chat/copilot-chat-input.component.mjs +666 -0
  19. package/dist/esm2022/components/chat/copilot-chat-input.types.mjs +10 -0
  20. package/dist/esm2022/components/chat/copilot-chat-message-view-cursor.component.mjs +45 -0
  21. package/dist/esm2022/components/chat/copilot-chat-message-view.component.mjs +296 -0
  22. package/dist/esm2022/components/chat/copilot-chat-message-view.types.mjs +2 -0
  23. package/dist/esm2022/components/chat/copilot-chat-textarea.component.mjs +188 -0
  24. package/dist/esm2022/components/chat/copilot-chat-tool-calls-view.component.mjs +216 -0
  25. package/dist/esm2022/components/chat/copilot-chat-toolbar.component.mjs +25 -0
  26. package/dist/esm2022/components/chat/copilot-chat-tools-menu.component.mjs +199 -0
  27. package/dist/esm2022/components/chat/copilot-chat-user-message-branch-navigation.component.mjs +137 -0
  28. package/dist/esm2022/components/chat/copilot-chat-user-message-buttons.component.mjs +207 -0
  29. package/dist/esm2022/components/chat/copilot-chat-user-message-renderer.component.mjs +35 -0
  30. package/dist/esm2022/components/chat/copilot-chat-user-message-toolbar.component.mjs +34 -0
  31. package/dist/esm2022/components/chat/copilot-chat-user-message.component.mjs +341 -0
  32. package/dist/esm2022/components/chat/copilot-chat-user-message.types.mjs +2 -0
  33. package/dist/esm2022/components/chat/copilot-chat-view-disclaimer.component.mjs +52 -0
  34. package/dist/esm2022/components/chat/copilot-chat-view-feather.component.mjs +55 -0
  35. package/dist/esm2022/components/chat/copilot-chat-view-handlers.service.mjs +19 -0
  36. package/dist/esm2022/components/chat/copilot-chat-view-input-container.component.mjs +110 -0
  37. package/dist/esm2022/components/chat/copilot-chat-view-scroll-to-bottom-button.component.mjs +93 -0
  38. package/dist/esm2022/components/chat/copilot-chat-view-scroll-view.component.mjs +443 -0
  39. package/dist/esm2022/components/chat/copilot-chat-view.component.mjs +479 -0
  40. package/dist/esm2022/components/chat/copilot-chat-view.types.mjs +2 -0
  41. package/dist/esm2022/components/chat/copilot-chat.component.mjs +214 -0
  42. package/dist/esm2022/components/copilotkit-tool-render.component.mjs +153 -0
  43. package/dist/esm2022/copilotkitnext-angular.mjs +5 -0
  44. package/dist/esm2022/core/chat-configuration/chat-configuration.providers.mjs +65 -0
  45. package/dist/esm2022/core/chat-configuration/chat-configuration.service.mjs +145 -0
  46. package/dist/esm2022/core/chat-configuration/chat-configuration.types.mjs +26 -0
  47. package/dist/esm2022/core/copilotkit.providers.mjs +34 -0
  48. package/dist/esm2022/core/copilotkit.service.mjs +426 -0
  49. package/dist/esm2022/core/copilotkit.types.mjs +13 -0
  50. package/dist/esm2022/directives/copilotkit-agent-context.directive.mjs +130 -0
  51. package/dist/esm2022/directives/copilotkit-agent.directive.mjs +217 -0
  52. package/dist/esm2022/directives/copilotkit-chat-config.directive.mjs +218 -0
  53. package/dist/esm2022/directives/copilotkit-config.directive.mjs +94 -0
  54. package/dist/esm2022/directives/copilotkit-frontend-tool.directive.mjs +128 -0
  55. package/dist/esm2022/directives/copilotkit-human-in-the-loop.directive.mjs +265 -0
  56. package/dist/esm2022/directives/stick-to-bottom.directive.mjs +181 -0
  57. package/dist/esm2022/index.mjs +70 -0
  58. package/dist/esm2022/lib/directives/tooltip.directive.mjs +211 -0
  59. package/dist/esm2022/lib/slots/copilot-slot.component.mjs +144 -0
  60. package/dist/esm2022/lib/slots/slot.types.mjs +6 -0
  61. package/dist/esm2022/lib/slots/slot.utils.mjs +222 -0
  62. package/dist/esm2022/lib/utils.mjs +10 -0
  63. package/dist/esm2022/services/resize-observer.service.mjs +152 -0
  64. package/dist/esm2022/services/scroll-position.service.mjs +124 -0
  65. package/dist/esm2022/types/frontend-tool.mjs +2 -0
  66. package/dist/esm2022/types/human-in-the-loop.mjs +2 -0
  67. package/dist/esm2022/utils/agent-context.utils.mjs +114 -0
  68. package/dist/esm2022/utils/agent.utils.mjs +204 -0
  69. package/dist/esm2022/utils/chat-config.utils.mjs +186 -0
  70. package/dist/esm2022/utils/copilotkit.utils.mjs +20 -0
  71. package/dist/esm2022/utils/frontend-tool.utils.mjs +224 -0
  72. package/dist/esm2022/utils/human-in-the-loop.utils.mjs +293 -0
  73. package/dist/fesm2022/copilotkitnext-angular.mjs +174 -187
  74. package/dist/fesm2022/copilotkitnext-angular.mjs.map +1 -1
  75. package/dist/utils/frontend-tool.utils.d.ts +1 -1
  76. package/package.json +23 -20
  77. package/vitest.config.mts +32 -21
  78. package/.turbo/turbo-build.log +0 -38
  79. package/.turbo/turbo-check-types.log +0 -0
  80. package/.turbo/turbo-test.log +0 -71
  81. package/ng-package.json +0 -19
  82. package/src/components/chat/__tests__/copilot-chat-assistant-message.component.spec.ts +0 -282
  83. package/src/components/chat/__tests__/copilot-chat-input.component.spec.ts +0 -419
  84. package/src/components/chat/__tests__/copilot-chat-message-view.component.spec.ts +0 -372
  85. package/src/components/chat/__tests__/copilot-chat-user-message.component.spec.ts +0 -249
  86. package/src/components/chat/copilot-chat-assistant-message-buttons.component.ts +0 -292
  87. package/src/components/chat/copilot-chat-assistant-message-renderer.component.ts +0 -472
  88. package/src/components/chat/copilot-chat-assistant-message-toolbar.component.ts +0 -29
  89. package/src/components/chat/copilot-chat-assistant-message.component.ts +0 -463
  90. package/src/components/chat/copilot-chat-assistant-message.types.ts +0 -50
  91. package/src/components/chat/copilot-chat-audio-recorder.component.ts +0 -241
  92. package/src/components/chat/copilot-chat-buttons.component.ts +0 -308
  93. package/src/components/chat/copilot-chat-buttons.component.ts.bak +0 -471
  94. package/src/components/chat/copilot-chat-input-defaults.ts +0 -47
  95. package/src/components/chat/copilot-chat-input.component.ts +0 -512
  96. package/src/components/chat/copilot-chat-input.types.ts +0 -148
  97. package/src/components/chat/copilot-chat-message-view-cursor.component.ts +0 -51
  98. package/src/components/chat/copilot-chat-message-view.component.ts +0 -233
  99. package/src/components/chat/copilot-chat-message-view.types.ts +0 -39
  100. package/src/components/chat/copilot-chat-textarea.component.ts +0 -220
  101. package/src/components/chat/copilot-chat-tool-calls-view.component.ts +0 -261
  102. package/src/components/chat/copilot-chat-toolbar.component.ts +0 -35
  103. package/src/components/chat/copilot-chat-tools-menu.component.ts +0 -185
  104. package/src/components/chat/copilot-chat-user-message-branch-navigation.component.ts +0 -121
  105. package/src/components/chat/copilot-chat-user-message-buttons.component.ts +0 -170
  106. package/src/components/chat/copilot-chat-user-message-renderer.component.ts +0 -37
  107. package/src/components/chat/copilot-chat-user-message-toolbar.component.ts +0 -37
  108. package/src/components/chat/copilot-chat-user-message.component.ts +0 -247
  109. package/src/components/chat/copilot-chat-user-message.types.ts +0 -42
  110. package/src/components/chat/copilot-chat-view-disclaimer.component.ts +0 -51
  111. package/src/components/chat/copilot-chat-view-feather.component.ts +0 -47
  112. package/src/components/chat/copilot-chat-view-handlers.service.ts +0 -14
  113. package/src/components/chat/copilot-chat-view-input-container.component.ts +0 -87
  114. package/src/components/chat/copilot-chat-view-scroll-to-bottom-button.component.ts +0 -79
  115. package/src/components/chat/copilot-chat-view-scroll-view.component.ts +0 -322
  116. package/src/components/chat/copilot-chat-view.component.ts +0 -420
  117. package/src/components/chat/copilot-chat-view.types.ts +0 -52
  118. package/src/components/chat/copilot-chat.component.ts +0 -232
  119. package/src/components/copilotkit-tool-render.component.ts +0 -169
  120. package/src/core/__tests__/copilotkit.service.spec.ts +0 -1051
  121. package/src/core/__tests__/copilotkit.service.wildcard.spec.ts +0 -316
  122. package/src/core/chat-configuration/__tests__/chat-configuration.service.spec.ts +0 -287
  123. package/src/core/chat-configuration/chat-configuration.providers.ts +0 -71
  124. package/src/core/chat-configuration/chat-configuration.service.ts +0 -162
  125. package/src/core/chat-configuration/chat-configuration.types.ts +0 -57
  126. package/src/core/copilotkit.providers.ts +0 -59
  127. package/src/core/copilotkit.service.ts +0 -542
  128. package/src/core/copilotkit.types.ts +0 -132
  129. package/src/directives/__tests__/copilotkit-agent-context.directive.spec.ts +0 -384
  130. package/src/directives/__tests__/copilotkit-agent.directive.spec.ts +0 -253
  131. package/src/directives/__tests__/copilotkit-chat-config.directive.spec.ts +0 -385
  132. package/src/directives/__tests__/copilotkit-config.directive.spec.ts +0 -69
  133. package/src/directives/__tests__/copilotkit-frontend-tool-simple.directive.spec.ts +0 -60
  134. package/src/directives/__tests__/copilotkit-frontend-tool.directive.spec.ts +0 -108
  135. package/src/directives/__tests__/copilotkit-human-in-the-loop.directive.spec.ts +0 -452
  136. package/src/directives/copilotkit-agent-context.directive.ts +0 -138
  137. package/src/directives/copilotkit-agent.directive.ts +0 -225
  138. package/src/directives/copilotkit-chat-config.directive.ts +0 -241
  139. package/src/directives/copilotkit-config.directive.ts +0 -81
  140. package/src/directives/copilotkit-frontend-tool.directive.ts +0 -145
  141. package/src/directives/copilotkit-human-in-the-loop.directive.ts +0 -281
  142. package/src/directives/stick-to-bottom.directive.ts +0 -204
  143. package/src/index.ts +0 -105
  144. package/src/lib/directives/tooltip.directive.ts +0 -292
  145. package/src/lib/slots/__tests__/slot.utils.spec.ts +0 -377
  146. package/src/lib/slots/copilot-slot.component.ts +0 -135
  147. package/src/lib/slots/index.ts +0 -3
  148. package/src/lib/slots/slot.types.ts +0 -64
  149. package/src/lib/slots/slot.utils.ts +0 -289
  150. package/src/lib/utils.ts +0 -10
  151. package/src/public-api.ts +0 -1
  152. package/src/services/resize-observer.service.ts +0 -181
  153. package/src/services/scroll-position.service.ts +0 -169
  154. package/src/styles/globals.css +0 -266
  155. package/src/styles/index.css +0 -3
  156. package/src/test-setup.ts +0 -15
  157. package/src/testing/index.ts +0 -3
  158. package/src/testing/testing.utils.ts +0 -248
  159. package/src/types/frontend-tool.ts +0 -44
  160. package/src/types/human-in-the-loop.ts +0 -52
  161. package/src/utils/__tests__/agent.utils.spec.ts +0 -234
  162. package/src/utils/__tests__/chat-config.utils.spec.ts +0 -306
  163. package/src/utils/__tests__/frontend-tool-inject.spec.ts +0 -350
  164. package/src/utils/__tests__/frontend-tool-integration.spec.ts +0 -199
  165. package/src/utils/__tests__/frontend-tool.utils.spec.ts +0 -272
  166. package/src/utils/__tests__/human-in-the-loop.utils.spec.ts +0 -365
  167. package/src/utils/agent-context.utils.ts +0 -133
  168. package/src/utils/agent.utils.ts +0 -239
  169. package/src/utils/chat-config.utils.ts +0 -221
  170. package/src/utils/copilotkit.utils.ts +0 -20
  171. package/src/utils/frontend-tool.utils.ts +0 -266
  172. package/src/utils/human-in-the-loop.utils.ts +0 -359
  173. package/tsconfig.spec.json +0 -12
@@ -0,0 +1,94 @@
1
+ import { Directive, Input, Inject } from "@angular/core";
2
+ import { CopilotKitService } from "../core/copilotkit.service";
3
+ import * as i0 from "@angular/core";
4
+ import * as i1 from "../core/copilotkit.service";
5
+ /**
6
+ * Directive to configure CopilotKit runtime settings declaratively in templates.
7
+ *
8
+ * @example
9
+ * ```html
10
+ * <div [copilotkitConfig]="{
11
+ * runtimeUrl: 'https://api.example.com',
12
+ * headers: { 'Authorization': 'Bearer token' }
13
+ * }">
14
+ * <!-- Your app content -->
15
+ * </div>
16
+ * ```
17
+ *
18
+ * Or with individual inputs:
19
+ * ```html
20
+ * <div copilotkitConfig
21
+ * [runtimeUrl]="apiUrl"
22
+ * [headers]="authHeaders"
23
+ * [agents]="myAgents">
24
+ * <!-- Your app content -->
25
+ * </div>
26
+ * ```
27
+ */
28
+ export class CopilotKitConfigDirective {
29
+ copilotkit;
30
+ constructor(copilotkit) {
31
+ this.copilotkit = copilotkit;
32
+ }
33
+ copilotkitConfig;
34
+ runtimeUrl;
35
+ headers;
36
+ properties;
37
+ agents;
38
+ ngOnChanges(changes) {
39
+ // Handle combined config object
40
+ if (changes['copilotkitConfig']) {
41
+ const config = this.copilotkitConfig;
42
+ if (config) {
43
+ if (config.runtimeUrl !== undefined) {
44
+ this.copilotkit.setRuntimeUrl(config.runtimeUrl);
45
+ }
46
+ if (config.headers) {
47
+ this.copilotkit.setHeaders(config.headers);
48
+ }
49
+ if (config.properties) {
50
+ this.copilotkit.setProperties(config.properties);
51
+ }
52
+ if (config.agents) {
53
+ this.copilotkit.setAgents(config.agents);
54
+ }
55
+ }
56
+ }
57
+ // Handle individual inputs
58
+ if (changes['runtimeUrl'] && !this.copilotkitConfig) {
59
+ this.copilotkit.setRuntimeUrl(this.runtimeUrl);
60
+ }
61
+ if (changes['headers'] && !this.copilotkitConfig) {
62
+ this.copilotkit.setHeaders(this.headers || {});
63
+ }
64
+ if (changes['properties'] && !this.copilotkitConfig) {
65
+ this.copilotkit.setProperties(this.properties || {});
66
+ }
67
+ if (changes['agents'] && !this.copilotkitConfig) {
68
+ this.copilotkit.setAgents(this.agents || {});
69
+ }
70
+ }
71
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CopilotKitConfigDirective, deps: [{ token: CopilotKitService }], target: i0.ɵɵFactoryTarget.Directive });
72
+ static ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "18.2.13", type: CopilotKitConfigDirective, isStandalone: true, selector: "[copilotkitConfig]", inputs: { copilotkitConfig: "copilotkitConfig", runtimeUrl: "runtimeUrl", headers: "headers", properties: "properties", agents: "agents" }, usesOnChanges: true, ngImport: i0 });
73
+ }
74
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CopilotKitConfigDirective, decorators: [{
75
+ type: Directive,
76
+ args: [{
77
+ selector: "[copilotkitConfig]",
78
+ standalone: true,
79
+ }]
80
+ }], ctorParameters: () => [{ type: i1.CopilotKitService, decorators: [{
81
+ type: Inject,
82
+ args: [CopilotKitService]
83
+ }] }], propDecorators: { copilotkitConfig: [{
84
+ type: Input
85
+ }], runtimeUrl: [{
86
+ type: Input
87
+ }], headers: [{
88
+ type: Input
89
+ }], properties: [{
90
+ type: Input
91
+ }], agents: [{
92
+ type: Input
93
+ }] } });
94
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29waWxvdGtpdC1jb25maWcuZGlyZWN0aXZlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL2RpcmVjdGl2ZXMvY29waWxvdGtpdC1jb25maWcuZGlyZWN0aXZlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxTQUFTLEVBQUUsS0FBSyxFQUE0QixNQUFNLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDbkYsT0FBTyxFQUFFLGlCQUFpQixFQUFFLE1BQU0sNEJBQTRCLENBQUM7OztBQUcvRDs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztHQXNCRztBQUtILE1BQU0sT0FBTyx5QkFBeUI7SUFDb0I7SUFBeEQsWUFBd0QsVUFBNkI7UUFBN0IsZUFBVSxHQUFWLFVBQVUsQ0FBbUI7SUFBRyxDQUFDO0lBRWhGLGdCQUFnQixDQUt2QjtJQUVPLFVBQVUsQ0FBVTtJQUNwQixPQUFPLENBQTBCO0lBQ2pDLFVBQVUsQ0FBMkI7SUFDckMsTUFBTSxDQUFpQztJQUVoRCxXQUFXLENBQUMsT0FBc0I7UUFDaEMsZ0NBQWdDO1FBQ2hDLElBQUksT0FBTyxDQUFDLGtCQUFrQixDQUFDLEVBQUUsQ0FBQztZQUNoQyxNQUFNLE1BQU0sR0FBRyxJQUFJLENBQUMsZ0JBQWdCLENBQUM7WUFDckMsSUFBSSxNQUFNLEVBQUUsQ0FBQztnQkFDWCxJQUFJLE1BQU0sQ0FBQyxVQUFVLEtBQUssU0FBUyxFQUFFLENBQUM7b0JBQ3BDLElBQUksQ0FBQyxVQUFVLENBQUMsYUFBYSxDQUFDLE1BQU0sQ0FBQyxVQUFVLENBQUMsQ0FBQztnQkFDbkQsQ0FBQztnQkFDRCxJQUFJLE1BQU0sQ0FBQyxPQUFPLEVBQUUsQ0FBQztvQkFDbkIsSUFBSSxDQUFDLFVBQVUsQ0FBQyxVQUFVLENBQUMsTUFBTSxDQUFDLE9BQU8sQ0FBQyxDQUFDO2dCQUM3QyxDQUFDO2dCQUNELElBQUksTUFBTSxDQUFDLFVBQVUsRUFBRSxDQUFDO29CQUN0QixJQUFJLENBQUMsVUFBVSxDQUFDLGFBQWEsQ0FBQyxNQUFNLENBQUMsVUFBVSxDQUFDLENBQUM7Z0JBQ25ELENBQUM7Z0JBQ0QsSUFBSSxNQUFNLENBQUMsTUFBTSxFQUFFLENBQUM7b0JBQ2xCLElBQUksQ0FBQyxVQUFVLENBQUMsU0FBUyxDQUFDLE1BQU0sQ0FBQyxNQUFNLENBQUMsQ0FBQztnQkFDM0MsQ0FBQztZQUNILENBQUM7UUFDSCxDQUFDO1FBRUQsMkJBQTJCO1FBQzNCLElBQUksT0FBTyxDQUFDLFlBQVksQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLGdCQUFnQixFQUFFLENBQUM7WUFDcEQsSUFBSSxDQUFDLFVBQVUsQ0FBQyxhQUFhLENBQUMsSUFBSSxDQUFDLFVBQVUsQ0FBQyxDQUFDO1FBQ2pELENBQUM7UUFDRCxJQUFJLE9BQU8sQ0FBQyxTQUFTLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxnQkFBZ0IsRUFBRSxDQUFDO1lBQ2pELElBQUksQ0FBQyxVQUFVLENBQUMsVUFBVSxDQUFDLElBQUksQ0FBQyxPQUFPLElBQUksRUFBRSxDQUFDLENBQUM7UUFDakQsQ0FBQztRQUNELElBQUksT0FBTyxDQUFDLFlBQVksQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLGdCQUFnQixFQUFFLENBQUM7WUFDcEQsSUFBSSxDQUFDLFVBQVUsQ0FBQyxhQUFhLENBQUMsSUFBSSxDQUFDLFVBQVUsSUFBSSxFQUFFLENBQUMsQ0FBQztRQUN2RCxDQUFDO1FBQ0QsSUFBSSxPQUFPLENBQUMsUUFBUSxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsZ0JBQWdCLEVBQUUsQ0FBQztZQUNoRCxJQUFJLENBQUMsVUFBVSxDQUFDLFNBQVMsQ0FBQyxJQUFJLENBQUMsTUFBTSxJQUFJLEVBQUUsQ0FBQyxDQUFDO1FBQy9DLENBQUM7SUFDSCxDQUFDO3dHQWhEVSx5QkFBeUIsa0JBQ2hCLGlCQUFpQjs0RkFEMUIseUJBQXlCOzs0RkFBekIseUJBQXlCO2tCQUpyQyxTQUFTO21CQUFDO29CQUNULFFBQVEsRUFBRSxvQkFBb0I7b0JBQzlCLFVBQVUsRUFBRSxJQUFJO2lCQUNqQjs7MEJBRWMsTUFBTTsyQkFBQyxpQkFBaUI7eUNBRTVCLGdCQUFnQjtzQkFBeEIsS0FBSztnQkFPRyxVQUFVO3NCQUFsQixLQUFLO2dCQUNHLE9BQU87c0JBQWYsS0FBSztnQkFDRyxVQUFVO3NCQUFsQixLQUFLO2dCQUNHLE1BQU07c0JBQWQsS0FBSyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IERpcmVjdGl2ZSwgSW5wdXQsIE9uQ2hhbmdlcywgU2ltcGxlQ2hhbmdlcywgSW5qZWN0IH0gZnJvbSBcIkBhbmd1bGFyL2NvcmVcIjtcbmltcG9ydCB7IENvcGlsb3RLaXRTZXJ2aWNlIH0gZnJvbSBcIi4uL2NvcmUvY29waWxvdGtpdC5zZXJ2aWNlXCI7XG5pbXBvcnQgeyBBYnN0cmFjdEFnZW50IH0gZnJvbSBcIkBhZy11aS9jbGllbnRcIjtcblxuLyoqXG4gKiBEaXJlY3RpdmUgdG8gY29uZmlndXJlIENvcGlsb3RLaXQgcnVudGltZSBzZXR0aW5ncyBkZWNsYXJhdGl2ZWx5IGluIHRlbXBsYXRlcy5cbiAqIFxuICogQGV4YW1wbGVcbiAqIGBgYGh0bWxcbiAqIDxkaXYgW2NvcGlsb3RraXRDb25maWddPVwie1xuICogICBydW50aW1lVXJsOiAnaHR0cHM6Ly9hcGkuZXhhbXBsZS5jb20nLFxuICogICBoZWFkZXJzOiB7ICdBdXRob3JpemF0aW9uJzogJ0JlYXJlciB0b2tlbicgfVxuICogfVwiPlxuICogICA8IS0tIFlvdXIgYXBwIGNvbnRlbnQgLS0+XG4gKiA8L2Rpdj5cbiAqIGBgYFxuICogXG4gKiBPciB3aXRoIGluZGl2aWR1YWwgaW5wdXRzOlxuICogYGBgaHRtbFxuICogPGRpdiBjb3BpbG90a2l0Q29uZmlnXG4gKiAgICAgIFtydW50aW1lVXJsXT1cImFwaVVybFwiXG4gKiAgICAgIFtoZWFkZXJzXT1cImF1dGhIZWFkZXJzXCJcbiAqICAgICAgW2FnZW50c109XCJteUFnZW50c1wiPlxuICogICA8IS0tIFlvdXIgYXBwIGNvbnRlbnQgLS0+XG4gKiA8L2Rpdj5cbiAqIGBgYFxuICovXG5ARGlyZWN0aXZlKHtcbiAgc2VsZWN0b3I6IFwiW2NvcGlsb3RraXRDb25maWddXCIsXG4gIHN0YW5kYWxvbmU6IHRydWUsXG59KVxuZXhwb3J0IGNsYXNzIENvcGlsb3RLaXRDb25maWdEaXJlY3RpdmUgaW1wbGVtZW50cyBPbkNoYW5nZXMge1xuICBjb25zdHJ1Y3RvcihASW5qZWN0KENvcGlsb3RLaXRTZXJ2aWNlKSBwcml2YXRlIHJlYWRvbmx5IGNvcGlsb3RraXQ6IENvcGlsb3RLaXRTZXJ2aWNlKSB7fVxuXG4gIEBJbnB1dCgpIGNvcGlsb3RraXRDb25maWc/OiB7XG4gICAgcnVudGltZVVybD86IHN0cmluZztcbiAgICBoZWFkZXJzPzogUmVjb3JkPHN0cmluZywgc3RyaW5nPjtcbiAgICBwcm9wZXJ0aWVzPzogUmVjb3JkPHN0cmluZywgdW5rbm93bj47XG4gICAgYWdlbnRzPzogUmVjb3JkPHN0cmluZywgQWJzdHJhY3RBZ2VudD47XG4gIH07XG5cbiAgQElucHV0KCkgcnVudGltZVVybD86IHN0cmluZztcbiAgQElucHV0KCkgaGVhZGVycz86IFJlY29yZDxzdHJpbmcsIHN0cmluZz47XG4gIEBJbnB1dCgpIHByb3BlcnRpZXM/OiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPjtcbiAgQElucHV0KCkgYWdlbnRzPzogUmVjb3JkPHN0cmluZywgQWJzdHJhY3RBZ2VudD47XG5cbiAgbmdPbkNoYW5nZXMoY2hhbmdlczogU2ltcGxlQ2hhbmdlcyk6IHZvaWQge1xuICAgIC8vIEhhbmRsZSBjb21iaW5lZCBjb25maWcgb2JqZWN0XG4gICAgaWYgKGNoYW5nZXNbJ2NvcGlsb3RraXRDb25maWcnXSkge1xuICAgICAgY29uc3QgY29uZmlnID0gdGhpcy5jb3BpbG90a2l0Q29uZmlnO1xuICAgICAgaWYgKGNvbmZpZykge1xuICAgICAgICBpZiAoY29uZmlnLnJ1bnRpbWVVcmwgIT09IHVuZGVmaW5lZCkge1xuICAgICAgICAgIHRoaXMuY29waWxvdGtpdC5zZXRSdW50aW1lVXJsKGNvbmZpZy5ydW50aW1lVXJsKTtcbiAgICAgICAgfVxuICAgICAgICBpZiAoY29uZmlnLmhlYWRlcnMpIHtcbiAgICAgICAgICB0aGlzLmNvcGlsb3RraXQuc2V0SGVhZGVycyhjb25maWcuaGVhZGVycyk7XG4gICAgICAgIH1cbiAgICAgICAgaWYgKGNvbmZpZy5wcm9wZXJ0aWVzKSB7XG4gICAgICAgICAgdGhpcy5jb3BpbG90a2l0LnNldFByb3BlcnRpZXMoY29uZmlnLnByb3BlcnRpZXMpO1xuICAgICAgICB9XG4gICAgICAgIGlmIChjb25maWcuYWdlbnRzKSB7XG4gICAgICAgICAgdGhpcy5jb3BpbG90a2l0LnNldEFnZW50cyhjb25maWcuYWdlbnRzKTtcbiAgICAgICAgfVxuICAgICAgfVxuICAgIH1cblxuICAgIC8vIEhhbmRsZSBpbmRpdmlkdWFsIGlucHV0c1xuICAgIGlmIChjaGFuZ2VzWydydW50aW1lVXJsJ10gJiYgIXRoaXMuY29waWxvdGtpdENvbmZpZykge1xuICAgICAgdGhpcy5jb3BpbG90a2l0LnNldFJ1bnRpbWVVcmwodGhpcy5ydW50aW1lVXJsKTtcbiAgICB9XG4gICAgaWYgKGNoYW5nZXNbJ2hlYWRlcnMnXSAmJiAhdGhpcy5jb3BpbG90a2l0Q29uZmlnKSB7XG4gICAgICB0aGlzLmNvcGlsb3RraXQuc2V0SGVhZGVycyh0aGlzLmhlYWRlcnMgfHwge30pO1xuICAgIH1cbiAgICBpZiAoY2hhbmdlc1sncHJvcGVydGllcyddICYmICF0aGlzLmNvcGlsb3RraXRDb25maWcpIHtcbiAgICAgIHRoaXMuY29waWxvdGtpdC5zZXRQcm9wZXJ0aWVzKHRoaXMucHJvcGVydGllcyB8fCB7fSk7XG4gICAgfVxuICAgIGlmIChjaGFuZ2VzWydhZ2VudHMnXSAmJiAhdGhpcy5jb3BpbG90a2l0Q29uZmlnKSB7XG4gICAgICB0aGlzLmNvcGlsb3RraXQuc2V0QWdlbnRzKHRoaXMuYWdlbnRzIHx8IHt9KTtcbiAgICB9XG4gIH1cbn0iXX0=
@@ -0,0 +1,128 @@
1
+ import { Directive, Input, isDevMode, Inject, } from "@angular/core";
2
+ import { CopilotKitService } from "../core/copilotkit.service";
3
+ import * as i0 from "@angular/core";
4
+ import * as i1 from "../core/copilotkit.service";
5
+ export class CopilotKitFrontendToolDirective {
6
+ copilotkit;
7
+ isRegistered = false;
8
+ constructor(copilotkit) {
9
+ this.copilotkit = copilotkit;
10
+ }
11
+ name;
12
+ description;
13
+ parameters;
14
+ handler;
15
+ render;
16
+ followUp;
17
+ // Alternative: Accept a full tool object
18
+ tool;
19
+ ngOnInit() {
20
+ this.registerTool();
21
+ }
22
+ ngOnChanges(_changes) {
23
+ if (this.isRegistered) {
24
+ // Re-register the tool if any properties change
25
+ this.unregisterTool();
26
+ this.registerTool();
27
+ }
28
+ }
29
+ ngOnDestroy() {
30
+ this.unregisterTool();
31
+ }
32
+ registerTool() {
33
+ const tool = this.getTool();
34
+ if (!tool.name) {
35
+ if (isDevMode()) {
36
+ console.warn('CopilotKitFrontendToolDirective: "name" is required. ' +
37
+ 'Please provide a name via [name]="toolName" or ' +
38
+ "[copilotkitFrontendTool]=\"{ name: 'toolName', ... }\"");
39
+ }
40
+ return; // Don't register if no name
41
+ }
42
+ // Register the tool with CopilotKit
43
+ this.copilotkit.copilotkit.addTool(tool);
44
+ // Register the render if provided
45
+ if (tool.render) {
46
+ const currentRenders = this.copilotkit.currentRenderToolCalls();
47
+ const renderEntry = {
48
+ name: tool.name,
49
+ render: tool.render,
50
+ };
51
+ // Check for duplicate
52
+ const existingIndex = currentRenders.findIndex((r) => r.name === tool.name);
53
+ if (existingIndex !== -1) {
54
+ if (isDevMode()) {
55
+ console.warn(`[CopilotKit] Tool "${tool.name}" already has a render. ` +
56
+ `The previous render will be replaced. ` +
57
+ `This may indicate a duplicate tool registration.`);
58
+ }
59
+ const updated = [...currentRenders];
60
+ updated[existingIndex] = renderEntry;
61
+ this.copilotkit.setCurrentRenderToolCalls(updated);
62
+ }
63
+ else {
64
+ this.copilotkit.setCurrentRenderToolCalls([...currentRenders, renderEntry]);
65
+ }
66
+ }
67
+ this.isRegistered = true;
68
+ }
69
+ unregisterTool() {
70
+ if (!this.isRegistered)
71
+ return;
72
+ const tool = this.getTool();
73
+ if (tool.name) {
74
+ // Remove the tool
75
+ this.copilotkit.copilotkit.removeTool(tool.name);
76
+ // Remove the render if it exists
77
+ const currentRenders = this.copilotkit.currentRenderToolCalls();
78
+ const filtered = currentRenders.filter((r) => r.name !== tool.name);
79
+ if (filtered.length !== currentRenders.length) {
80
+ this.copilotkit.setCurrentRenderToolCalls(filtered);
81
+ }
82
+ }
83
+ this.isRegistered = false;
84
+ }
85
+ getTool() {
86
+ // If full tool object is provided, use it
87
+ if (this.tool) {
88
+ return this.tool;
89
+ }
90
+ // Otherwise, construct from individual inputs
91
+ return {
92
+ name: this.name,
93
+ description: this.description,
94
+ parameters: this.parameters,
95
+ handler: this.handler,
96
+ render: this.render,
97
+ followUp: this.followUp,
98
+ };
99
+ }
100
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CopilotKitFrontendToolDirective, deps: [{ token: CopilotKitService }], target: i0.ɵɵFactoryTarget.Directive });
101
+ static ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "18.2.13", type: CopilotKitFrontendToolDirective, isStandalone: true, selector: "[copilotkitFrontendTool]", inputs: { name: "name", description: "description", parameters: "parameters", handler: "handler", render: "render", followUp: "followUp", tool: ["copilotkitFrontendTool", "tool"] }, usesOnChanges: true, ngImport: i0 });
102
+ }
103
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CopilotKitFrontendToolDirective, decorators: [{
104
+ type: Directive,
105
+ args: [{
106
+ selector: "[copilotkitFrontendTool]",
107
+ standalone: true,
108
+ }]
109
+ }], ctorParameters: () => [{ type: i1.CopilotKitService, decorators: [{
110
+ type: Inject,
111
+ args: [CopilotKitService]
112
+ }] }], propDecorators: { name: [{
113
+ type: Input
114
+ }], description: [{
115
+ type: Input
116
+ }], parameters: [{
117
+ type: Input
118
+ }], handler: [{
119
+ type: Input
120
+ }], render: [{
121
+ type: Input
122
+ }], followUp: [{
123
+ type: Input
124
+ }], tool: [{
125
+ type: Input,
126
+ args: ["copilotkitFrontendTool"]
127
+ }] } });
128
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"copilotkit-frontend-tool.directive.js","sourceRoot":"","sources":["../../../src/directives/copilotkit-frontend-tool.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Directive,\n  Input,\n  OnInit,\n  OnChanges,\n  OnDestroy,\n  SimpleChanges,\n  TemplateRef,\n  Type,\n  isDevMode,\n  Inject,\n} from \"@angular/core\";\nimport { CopilotKitService } from \"../core/copilotkit.service\";\nimport type {\n  AngularFrontendTool,\n  AngularToolCallRender,\n  ToolCallRender,\n} from \"../core/copilotkit.types\";\nimport type { z } from \"zod\";\n\n@Directive({\n  selector: \"[copilotkitFrontendTool]\",\n  standalone: true,\n})\nexport class CopilotKitFrontendToolDirective<\n    T extends Record<string, any> = Record<string, any>,\n  >\n  implements OnInit, OnChanges, OnDestroy\n{\n  private isRegistered = false;\n\n  constructor(\n    @Inject(CopilotKitService) private readonly copilotkit: CopilotKitService\n  ) {}\n\n  @Input() name!: string;\n  @Input() description?: string;\n  @Input() parameters?: z.ZodSchema<T>;\n  @Input() handler?: (args: T) => Promise<any>;\n  @Input() render?: Type<any> | TemplateRef<any>;\n  @Input() followUp?: boolean;\n\n  // Alternative: Accept a full tool object\n  @Input(\"copilotkitFrontendTool\") tool?: AngularFrontendTool<T>;\n\n  ngOnInit(): void {\n    this.registerTool();\n  }\n\n  ngOnChanges(_changes: SimpleChanges): void {\n    if (this.isRegistered) {\n      // Re-register the tool if any properties change\n      this.unregisterTool();\n      this.registerTool();\n    }\n  }\n\n  ngOnDestroy(): void {\n    this.unregisterTool();\n  }\n\n  private registerTool(): void {\n    const tool = this.getTool();\n\n    if (!tool.name) {\n      if (isDevMode()) {\n        console.warn(\n          'CopilotKitFrontendToolDirective: \"name\" is required. ' +\n            'Please provide a name via [name]=\"toolName\" or ' +\n            \"[copilotkitFrontendTool]=\\\"{ name: 'toolName', ... }\\\"\"\n        );\n      }\n      return; // Don't register if no name\n    }\n\n    // Register the tool with CopilotKit\n    this.copilotkit.copilotkit.addTool(tool);\n\n    // Register the render if provided\n    if (tool.render) {\n      const currentRenders = this.copilotkit.currentRenderToolCalls();\n      const renderEntry: AngularToolCallRender = {\n        name: tool.name,\n        render: tool.render,\n      };\n\n      // Check for duplicate\n      const existingIndex = currentRenders.findIndex((r: ToolCallRender) => r.name === tool.name);\n      if (existingIndex !== -1) {\n        if (isDevMode()) {\n          console.warn(\n            `[CopilotKit] Tool \"${tool.name}\" already has a render. ` +\n              `The previous render will be replaced. ` +\n              `This may indicate a duplicate tool registration.`\n          );\n        }\n        const updated = [...currentRenders];\n        updated[existingIndex] = renderEntry;\n        this.copilotkit.setCurrentRenderToolCalls(updated);\n      } else {\n        this.copilotkit.setCurrentRenderToolCalls([...currentRenders, renderEntry]);\n      }\n    }\n\n    this.isRegistered = true;\n  }\n\n  private unregisterTool(): void {\n    if (!this.isRegistered) return;\n\n    const tool = this.getTool();\n\n    if (tool.name) {\n      // Remove the tool\n      this.copilotkit.copilotkit.removeTool(tool.name);\n\n      // Remove the render if it exists\n      const currentRenders = this.copilotkit.currentRenderToolCalls();\n      const filtered = currentRenders.filter((r: ToolCallRender) => r.name !== tool.name);\n      if (filtered.length !== currentRenders.length) {\n        this.copilotkit.setCurrentRenderToolCalls(filtered);\n      }\n    }\n\n    this.isRegistered = false;\n  }\n\n  private getTool(): AngularFrontendTool<T> {\n    // If full tool object is provided, use it\n    if (this.tool) {\n      return this.tool;\n    }\n\n    // Otherwise, construct from individual inputs\n    return {\n      name: this.name,\n      description: this.description,\n      parameters: this.parameters,\n      handler: this.handler,\n      render: this.render,\n      followUp: this.followUp,\n    };\n  }\n}\n"]}
@@ -0,0 +1,265 @@
1
+ import { Directive, Input, Output, EventEmitter, signal, isDevMode, Inject, } from "@angular/core";
2
+ import { CopilotKitService } from "../core/copilotkit.service";
3
+ import { ToolCallStatus } from "../core/copilotkit.types";
4
+ import * as i0 from "@angular/core";
5
+ import * as i1 from "../core/copilotkit.service";
6
+ /**
7
+ * Directive for declaratively creating human-in-the-loop tools.
8
+ * Provides reactive outputs for status changes and response events.
9
+ *
10
+ * @example
11
+ * ```html
12
+ * <!-- Basic usage -->
13
+ * <div copilotkitHumanInTheLoop
14
+ * [name]="'requireApproval'"
15
+ * [description]="'Requires user approval'"
16
+ * [args]="argsSchema"
17
+ * [render]="approvalComponent"
18
+ * (statusChange)="onStatusChange($event)"
19
+ * (responseProvided)="onResponse($event)">
20
+ * </div>
21
+ *
22
+ * <!-- With template -->
23
+ * <div copilotkitHumanInTheLoop
24
+ * [name]="'requireApproval'"
25
+ * [description]="'Requires user approval'"
26
+ * [args]="argsSchema"
27
+ * [render]="approvalTemplate"
28
+ * [(status)]="approvalStatus">
29
+ * </div>
30
+ *
31
+ * <ng-template #approvalTemplate let-props>
32
+ * <div *ngIf="props.status === 'executing'">
33
+ * <p>{{ props.args.action }}</p>
34
+ * <button (click)="props.respond('approved')">Approve</button>
35
+ * <button (click)="props.respond('rejected')">Reject</button>
36
+ * </div>
37
+ * </ng-template>
38
+ * ```
39
+ */
40
+ export class CopilotKitHumanInTheLoopDirective {
41
+ copilotkit;
42
+ toolId;
43
+ statusSignal = signal(ToolCallStatus.InProgress);
44
+ resolvePromise = null;
45
+ _status = ToolCallStatus.InProgress;
46
+ constructor(copilotkit) {
47
+ this.copilotkit = copilotkit;
48
+ }
49
+ /**
50
+ * The name of the human-in-the-loop tool.
51
+ */
52
+ name;
53
+ /**
54
+ * Description of what the tool does.
55
+ */
56
+ description;
57
+ /**
58
+ * Zod schema for the tool parameters.
59
+ */
60
+ parameters;
61
+ /**
62
+ * Component class or template to render for user interaction.
63
+ */
64
+ render;
65
+ /**
66
+ * Whether the tool should be registered (default: true).
67
+ */
68
+ enabled = true;
69
+ /**
70
+ * Alternative input using the directive selector.
71
+ * Allows: [copilotkitHumanInTheLoop]="config"
72
+ */
73
+ set config(value) {
74
+ if (value) {
75
+ if (value.name)
76
+ this.name = value.name;
77
+ if (value.description)
78
+ this.description = value.description;
79
+ if ("parameters" in value && value.parameters)
80
+ this.parameters = value.parameters;
81
+ if ("render" in value && value.render)
82
+ this.render = value.render;
83
+ }
84
+ }
85
+ /**
86
+ * Emits when the status changes.
87
+ */
88
+ statusChange = new EventEmitter();
89
+ /**
90
+ * Two-way binding for status.
91
+ */
92
+ get status() {
93
+ return this._status;
94
+ }
95
+ set status(value) {
96
+ // Input setter for two-way binding (though typically read-only)
97
+ this._status = value;
98
+ }
99
+ /**
100
+ * Emits when a response is provided by the user.
101
+ */
102
+ responseProvided = new EventEmitter();
103
+ /**
104
+ * Emits when the tool execution starts.
105
+ */
106
+ executionStarted = new EventEmitter();
107
+ /**
108
+ * Emits when the tool execution completes.
109
+ */
110
+ executionCompleted = new EventEmitter();
111
+ ngOnInit() {
112
+ if (this.enabled) {
113
+ this.registerTool();
114
+ }
115
+ }
116
+ ngOnChanges(changes) {
117
+ const relevantChanges = changes["name"] ||
118
+ changes["description"] ||
119
+ changes["args"] ||
120
+ changes["render"] ||
121
+ changes["enabled"];
122
+ if (relevantChanges && !relevantChanges.firstChange) {
123
+ // Re-register the tool with new configuration
124
+ this.unregisterTool();
125
+ if (this.enabled) {
126
+ this.registerTool();
127
+ }
128
+ }
129
+ }
130
+ ngOnDestroy() {
131
+ this.unregisterTool();
132
+ }
133
+ /**
134
+ * Programmatically trigger a response.
135
+ * Useful when the directive is used as a controller.
136
+ */
137
+ respond(result) {
138
+ this.handleResponse(result);
139
+ }
140
+ registerTool() {
141
+ if (!this.name || !this.description || !this.parameters || !this.render) {
142
+ if (isDevMode()) {
143
+ throw new Error("CopilotKitHumanInTheLoopDirective: Missing required inputs. " +
144
+ "Required: name, description, parameters, and render.");
145
+ }
146
+ return;
147
+ }
148
+ // Create handler that returns a Promise
149
+ const handler = async (args) => {
150
+ return new Promise((resolve) => {
151
+ this.updateStatus(ToolCallStatus.Executing);
152
+ this.resolvePromise = resolve;
153
+ this.executionStarted.emit(args);
154
+ });
155
+ };
156
+ // Create the frontend tool with enhanced render
157
+ const frontendTool = {
158
+ name: this.name,
159
+ description: this.description,
160
+ parameters: this.parameters,
161
+ handler,
162
+ render: this.render, // Will be enhanced by the render component
163
+ };
164
+ // Add the tool (returns void, so we use the tool name as ID)
165
+ this.copilotkit.copilotkit.addTool(frontendTool);
166
+ this.toolId = this.name;
167
+ // Register the render with respond capability
168
+ this.copilotkit.registerToolRender(this.name, {
169
+ name: this.name,
170
+ render: this.createEnhancedRender(),
171
+ });
172
+ }
173
+ unregisterTool() {
174
+ if (this.toolId) {
175
+ this.copilotkit.copilotkit.removeTool(this.toolId);
176
+ this.copilotkit.unregisterToolRender(this.name);
177
+ this.toolId = undefined;
178
+ }
179
+ }
180
+ createEnhancedRender() {
181
+ // If it's a template, we need to wrap it with our respond function
182
+ // This is handled by returning a special marker that the render component
183
+ // will recognize and enhance with the respond function
184
+ // Store reference to this directive instance for the render component
185
+ this.render.__humanInTheLoopDirective = this;
186
+ this.render.__humanInTheLoopStatus = this.statusSignal;
187
+ return this.render;
188
+ }
189
+ handleResponse(result) {
190
+ if (this.resolvePromise) {
191
+ this.resolvePromise(result);
192
+ this.updateStatus(ToolCallStatus.Complete);
193
+ this.resolvePromise = null;
194
+ this.responseProvided.emit(result);
195
+ this.executionCompleted.emit(result);
196
+ }
197
+ }
198
+ updateStatus(status) {
199
+ this._status = status;
200
+ this.statusSignal.set(status);
201
+ this.statusChange.emit(status);
202
+ }
203
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CopilotKitHumanInTheLoopDirective, deps: [{ token: CopilotKitService }], target: i0.ɵɵFactoryTarget.Directive });
204
+ static ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "18.2.13", type: CopilotKitHumanInTheLoopDirective, isStandalone: true, selector: "[copilotkitHumanInTheLoop]", inputs: { name: "name", description: "description", parameters: "parameters", render: "render", enabled: "enabled", config: ["copilotkitHumanInTheLoop", "config"], status: "status" }, outputs: { statusChange: "statusChange", responseProvided: "responseProvided", executionStarted: "executionStarted", executionCompleted: "executionCompleted" }, usesOnChanges: true, ngImport: i0 });
205
+ }
206
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CopilotKitHumanInTheLoopDirective, decorators: [{
207
+ type: Directive,
208
+ args: [{
209
+ selector: "[copilotkitHumanInTheLoop]",
210
+ standalone: true,
211
+ }]
212
+ }], ctorParameters: () => [{ type: i1.CopilotKitService, decorators: [{
213
+ type: Inject,
214
+ args: [CopilotKitService]
215
+ }] }], propDecorators: { name: [{
216
+ type: Input
217
+ }], description: [{
218
+ type: Input
219
+ }], parameters: [{
220
+ type: Input
221
+ }], render: [{
222
+ type: Input
223
+ }], enabled: [{
224
+ type: Input
225
+ }], config: [{
226
+ type: Input,
227
+ args: ["copilotkitHumanInTheLoop"]
228
+ }], statusChange: [{
229
+ type: Output
230
+ }], status: [{
231
+ type: Input
232
+ }], responseProvided: [{
233
+ type: Output
234
+ }], executionStarted: [{
235
+ type: Output
236
+ }], executionCompleted: [{
237
+ type: Output
238
+ }] } });
239
+ /**
240
+ * Helper directive to provide respond function in templates.
241
+ * This would be used internally by the tool render component.
242
+ *
243
+ * @internal
244
+ */
245
+ export class CopilotKitHumanInTheLoopRespondDirective {
246
+ copilotkitHumanInTheLoopRespond;
247
+ /**
248
+ * Convenience method for templates to call respond.
249
+ */
250
+ respond(result) {
251
+ this.copilotkitHumanInTheLoopRespond?.(result);
252
+ }
253
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CopilotKitHumanInTheLoopRespondDirective, deps: [], target: i0.ɵɵFactoryTarget.Directive });
254
+ static ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "18.2.13", type: CopilotKitHumanInTheLoopRespondDirective, isStandalone: true, selector: "[copilotkitHumanInTheLoopRespond]", inputs: { copilotkitHumanInTheLoopRespond: "copilotkitHumanInTheLoopRespond" }, ngImport: i0 });
255
+ }
256
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CopilotKitHumanInTheLoopRespondDirective, decorators: [{
257
+ type: Directive,
258
+ args: [{
259
+ selector: "[copilotkitHumanInTheLoopRespond]",
260
+ standalone: true,
261
+ }]
262
+ }], propDecorators: { copilotkitHumanInTheLoopRespond: [{
263
+ type: Input
264
+ }] } });
265
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"copilotkit-human-in-the-loop.directive.js","sourceRoot":"","sources":["../../../src/directives/copilotkit-human-in-the-loop.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Directive,\n  Input,\n  Output,\n  EventEmitter,\n  OnInit,\n  OnChanges,\n  OnDestroy,\n  SimpleChanges,\n  TemplateRef,\n  Type,\n  signal,\n  isDevMode,\n  Inject,\n} from \"@angular/core\";\nimport { CopilotKitService } from \"../core/copilotkit.service\";\nimport type {\n  AngularHumanInTheLoop,\n  HumanInTheLoopProps,\n  AngularFrontendTool,\n} from \"../core/copilotkit.types\";\nimport { ToolCallStatus } from \"../core/copilotkit.types\";\nimport * as z from \"zod\";\n\n/**\n * Directive for declaratively creating human-in-the-loop tools.\n * Provides reactive outputs for status changes and response events.\n *\n * @example\n * ```html\n * <!-- Basic usage -->\n * <div copilotkitHumanInTheLoop\n *      [name]=\"'requireApproval'\"\n *      [description]=\"'Requires user approval'\"\n *      [args]=\"argsSchema\"\n *      [render]=\"approvalComponent\"\n *      (statusChange)=\"onStatusChange($event)\"\n *      (responseProvided)=\"onResponse($event)\">\n * </div>\n *\n * <!-- With template -->\n * <div copilotkitHumanInTheLoop\n *      [name]=\"'requireApproval'\"\n *      [description]=\"'Requires user approval'\"\n *      [args]=\"argsSchema\"\n *      [render]=\"approvalTemplate\"\n *      [(status)]=\"approvalStatus\">\n * </div>\n *\n * <ng-template #approvalTemplate let-props>\n *   <div *ngIf=\"props.status === 'executing'\">\n *     <p>{{ props.args.action }}</p>\n *     <button (click)=\"props.respond('approved')\">Approve</button>\n *     <button (click)=\"props.respond('rejected')\">Reject</button>\n *   </div>\n * </ng-template>\n * ```\n */\n@Directive({\n  selector: \"[copilotkitHumanInTheLoop]\",\n  standalone: true,\n})\nexport class CopilotKitHumanInTheLoopDirective<\n    T extends Record<string, any> = Record<string, any>,\n  >\n  implements OnInit, OnChanges, OnDestroy\n{\n  private toolId?: string;\n  private statusSignal = signal<ToolCallStatus>(ToolCallStatus.InProgress);\n  private resolvePromise: ((result: unknown) => void) | null = null;\n  private _status: ToolCallStatus = ToolCallStatus.InProgress;\n\n  constructor(\n    @Inject(CopilotKitService) private readonly copilotkit: CopilotKitService\n  ) {}\n\n  /**\n   * The name of the human-in-the-loop tool.\n   */\n  @Input() name!: string;\n\n  /**\n   * Description of what the tool does.\n   */\n  @Input() description!: string;\n\n  /**\n   * Zod schema for the tool parameters.\n   */\n  @Input() parameters!: z.ZodSchema<T>;\n\n  /**\n   * Component class or template to render for user interaction.\n   */\n  @Input() render!: Type<any> | TemplateRef<HumanInTheLoopProps<T>>;\n\n  /**\n   * Whether the tool should be registered (default: true).\n   */\n  @Input() enabled = true;\n\n  /**\n   * Alternative input using the directive selector.\n   * Allows: [copilotkitHumanInTheLoop]=\"config\"\n   */\n  @Input(\"copilotkitHumanInTheLoop\")\n  set config(value: Partial<AngularHumanInTheLoop<T>> | undefined) {\n    if (value) {\n      if (value.name) this.name = value.name;\n      if (value.description) this.description = value.description;\n      if (\"parameters\" in value && value.parameters)\n        this.parameters = value.parameters as z.ZodSchema<T>;\n      if (\"render\" in value && value.render) this.render = value.render;\n    }\n  }\n\n  /**\n   * Emits when the status changes.\n   */\n  @Output() statusChange = new EventEmitter<ToolCallStatus>();\n\n  /**\n   * Two-way binding for status.\n   */\n  @Input()\n  get status(): ToolCallStatus {\n    return this._status;\n  }\n  set status(value: ToolCallStatus) {\n    // Input setter for two-way binding (though typically read-only)\n    this._status = value;\n  }\n\n  /**\n   * Emits when a response is provided by the user.\n   */\n  @Output() responseProvided = new EventEmitter<unknown>();\n\n  /**\n   * Emits when the tool execution starts.\n   */\n  @Output() executionStarted = new EventEmitter<any>();\n\n  /**\n   * Emits when the tool execution completes.\n   */\n  @Output() executionCompleted = new EventEmitter<unknown>();\n\n  ngOnInit(): void {\n    if (this.enabled) {\n      this.registerTool();\n    }\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    const relevantChanges =\n      changes[\"name\"] ||\n      changes[\"description\"] ||\n      changes[\"args\"] ||\n      changes[\"render\"] ||\n      changes[\"enabled\"];\n\n    if (relevantChanges && !relevantChanges.firstChange) {\n      // Re-register the tool with new configuration\n      this.unregisterTool();\n      if (this.enabled) {\n        this.registerTool();\n      }\n    }\n  }\n\n  ngOnDestroy(): void {\n    this.unregisterTool();\n  }\n\n  /**\n   * Programmatically trigger a response.\n   * Useful when the directive is used as a controller.\n   */\n  respond(result: unknown): void {\n    this.handleResponse(result);\n  }\n\n  private registerTool(): void {\n    if (!this.name || !this.description || !this.parameters || !this.render) {\n      if (isDevMode()) {\n        throw new Error(\n          \"CopilotKitHumanInTheLoopDirective: Missing required inputs. \" +\n            \"Required: name, description, parameters, and render.\"\n        );\n      }\n      return;\n    }\n\n    // Create handler that returns a Promise\n    const handler = async (args: T): Promise<unknown> => {\n      return new Promise((resolve) => {\n        this.updateStatus(ToolCallStatus.Executing);\n        this.resolvePromise = resolve;\n        this.executionStarted.emit(args);\n      });\n    };\n\n    // Create the frontend tool with enhanced render\n    const frontendTool: AngularFrontendTool<T> = {\n      name: this.name,\n      description: this.description,\n      parameters: this.parameters,\n      handler,\n      render: this.render, // Will be enhanced by the render component\n    };\n\n    // Add the tool (returns void, so we use the tool name as ID)\n    this.copilotkit.copilotkit.addTool(frontendTool);\n    this.toolId = this.name;\n\n    // Register the render with respond capability\n    this.copilotkit.registerToolRender(this.name, {\n      name: this.name,\n      render: this.createEnhancedRender(),\n    });\n  }\n\n  private unregisterTool(): void {\n    if (this.toolId) {\n      this.copilotkit.copilotkit.removeTool(this.toolId);\n      this.copilotkit.unregisterToolRender(this.name);\n      this.toolId = undefined;\n    }\n  }\n\n  private createEnhancedRender(): Type<any> | TemplateRef<any> {\n    // If it's a template, we need to wrap it with our respond function\n    // This is handled by returning a special marker that the render component\n    // will recognize and enhance with the respond function\n\n    // Store reference to this directive instance for the render component\n    (this.render as any).__humanInTheLoopDirective = this;\n    (this.render as any).__humanInTheLoopStatus = this.statusSignal;\n\n    return this.render;\n  }\n\n  private handleResponse(result: unknown): void {\n    if (this.resolvePromise) {\n      this.resolvePromise(result);\n      this.updateStatus(ToolCallStatus.Complete);\n      this.resolvePromise = null;\n      this.responseProvided.emit(result);\n      this.executionCompleted.emit(result);\n    }\n  }\n\n  private updateStatus(status: ToolCallStatus): void {\n    this._status = status;\n    this.statusSignal.set(status);\n    this.statusChange.emit(status);\n  }\n}\n\n/**\n * Helper directive to provide respond function in templates.\n * This would be used internally by the tool render component.\n *\n * @internal\n */\n@Directive({\n  selector: \"[copilotkitHumanInTheLoopRespond]\",\n  standalone: true,\n})\nexport class CopilotKitHumanInTheLoopRespondDirective {\n  @Input() copilotkitHumanInTheLoopRespond?: (result: unknown) => Promise<void>;\n\n  /**\n   * Convenience method for templates to call respond.\n   */\n  respond(result: unknown): void {\n    this.copilotkitHumanInTheLoopRespond?.(result);\n  }\n}\n"]}