@copilotkitnext/angular 0.0.2 → 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 (168) 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/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/package.json +21 -18
  72. package/vitest.config.mts +32 -21
  73. package/.turbo/turbo-build.log +0 -38
  74. package/.turbo/turbo-check-types.log +0 -0
  75. package/.turbo/turbo-test.log +0 -71
  76. package/ng-package.json +0 -19
  77. package/src/components/chat/__tests__/copilot-chat-assistant-message.component.spec.ts +0 -282
  78. package/src/components/chat/__tests__/copilot-chat-input.component.spec.ts +0 -419
  79. package/src/components/chat/__tests__/copilot-chat-message-view.component.spec.ts +0 -372
  80. package/src/components/chat/__tests__/copilot-chat-user-message.component.spec.ts +0 -249
  81. package/src/components/chat/copilot-chat-assistant-message-buttons.component.ts +0 -292
  82. package/src/components/chat/copilot-chat-assistant-message-renderer.component.ts +0 -472
  83. package/src/components/chat/copilot-chat-assistant-message-toolbar.component.ts +0 -29
  84. package/src/components/chat/copilot-chat-assistant-message.component.ts +0 -463
  85. package/src/components/chat/copilot-chat-assistant-message.types.ts +0 -50
  86. package/src/components/chat/copilot-chat-audio-recorder.component.ts +0 -241
  87. package/src/components/chat/copilot-chat-buttons.component.ts +0 -308
  88. package/src/components/chat/copilot-chat-buttons.component.ts.bak +0 -471
  89. package/src/components/chat/copilot-chat-input-defaults.ts +0 -47
  90. package/src/components/chat/copilot-chat-input.component.ts +0 -512
  91. package/src/components/chat/copilot-chat-input.types.ts +0 -148
  92. package/src/components/chat/copilot-chat-message-view-cursor.component.ts +0 -51
  93. package/src/components/chat/copilot-chat-message-view.component.ts +0 -233
  94. package/src/components/chat/copilot-chat-message-view.types.ts +0 -39
  95. package/src/components/chat/copilot-chat-textarea.component.ts +0 -220
  96. package/src/components/chat/copilot-chat-tool-calls-view.component.ts +0 -261
  97. package/src/components/chat/copilot-chat-toolbar.component.ts +0 -35
  98. package/src/components/chat/copilot-chat-tools-menu.component.ts +0 -185
  99. package/src/components/chat/copilot-chat-user-message-branch-navigation.component.ts +0 -121
  100. package/src/components/chat/copilot-chat-user-message-buttons.component.ts +0 -170
  101. package/src/components/chat/copilot-chat-user-message-renderer.component.ts +0 -37
  102. package/src/components/chat/copilot-chat-user-message-toolbar.component.ts +0 -37
  103. package/src/components/chat/copilot-chat-user-message.component.ts +0 -247
  104. package/src/components/chat/copilot-chat-user-message.types.ts +0 -42
  105. package/src/components/chat/copilot-chat-view-disclaimer.component.ts +0 -51
  106. package/src/components/chat/copilot-chat-view-feather.component.ts +0 -47
  107. package/src/components/chat/copilot-chat-view-handlers.service.ts +0 -14
  108. package/src/components/chat/copilot-chat-view-input-container.component.ts +0 -87
  109. package/src/components/chat/copilot-chat-view-scroll-to-bottom-button.component.ts +0 -79
  110. package/src/components/chat/copilot-chat-view-scroll-view.component.ts +0 -322
  111. package/src/components/chat/copilot-chat-view.component.ts +0 -420
  112. package/src/components/chat/copilot-chat-view.types.ts +0 -52
  113. package/src/components/chat/copilot-chat.component.ts +0 -232
  114. package/src/components/copilotkit-tool-render.component.ts +0 -169
  115. package/src/core/__tests__/copilotkit.service.spec.ts +0 -1051
  116. package/src/core/__tests__/copilotkit.service.wildcard.spec.ts +0 -316
  117. package/src/core/chat-configuration/__tests__/chat-configuration.service.spec.ts +0 -287
  118. package/src/core/chat-configuration/chat-configuration.providers.ts +0 -71
  119. package/src/core/chat-configuration/chat-configuration.service.ts +0 -162
  120. package/src/core/chat-configuration/chat-configuration.types.ts +0 -57
  121. package/src/core/copilotkit.providers.ts +0 -59
  122. package/src/core/copilotkit.service.ts +0 -542
  123. package/src/core/copilotkit.types.ts +0 -132
  124. package/src/directives/__tests__/copilotkit-agent-context.directive.spec.ts +0 -384
  125. package/src/directives/__tests__/copilotkit-agent.directive.spec.ts +0 -253
  126. package/src/directives/__tests__/copilotkit-chat-config.directive.spec.ts +0 -385
  127. package/src/directives/__tests__/copilotkit-config.directive.spec.ts +0 -69
  128. package/src/directives/__tests__/copilotkit-frontend-tool-simple.directive.spec.ts +0 -60
  129. package/src/directives/__tests__/copilotkit-frontend-tool.directive.spec.ts +0 -108
  130. package/src/directives/__tests__/copilotkit-human-in-the-loop.directive.spec.ts +0 -452
  131. package/src/directives/copilotkit-agent-context.directive.ts +0 -138
  132. package/src/directives/copilotkit-agent.directive.ts +0 -225
  133. package/src/directives/copilotkit-chat-config.directive.ts +0 -241
  134. package/src/directives/copilotkit-config.directive.ts +0 -81
  135. package/src/directives/copilotkit-frontend-tool.directive.ts +0 -145
  136. package/src/directives/copilotkit-human-in-the-loop.directive.ts +0 -281
  137. package/src/directives/stick-to-bottom.directive.ts +0 -204
  138. package/src/index.ts +0 -105
  139. package/src/lib/directives/tooltip.directive.ts +0 -292
  140. package/src/lib/slots/__tests__/slot.utils.spec.ts +0 -377
  141. package/src/lib/slots/copilot-slot.component.ts +0 -135
  142. package/src/lib/slots/index.ts +0 -3
  143. package/src/lib/slots/slot.types.ts +0 -64
  144. package/src/lib/slots/slot.utils.ts +0 -289
  145. package/src/lib/utils.ts +0 -10
  146. package/src/public-api.ts +0 -1
  147. package/src/services/resize-observer.service.ts +0 -181
  148. package/src/services/scroll-position.service.ts +0 -169
  149. package/src/styles/globals.css +0 -266
  150. package/src/styles/index.css +0 -3
  151. package/src/test-setup.ts +0 -15
  152. package/src/testing/index.ts +0 -3
  153. package/src/testing/testing.utils.ts +0 -248
  154. package/src/types/frontend-tool.ts +0 -44
  155. package/src/types/human-in-the-loop.ts +0 -52
  156. package/src/utils/__tests__/agent.utils.spec.ts +0 -234
  157. package/src/utils/__tests__/chat-config.utils.spec.ts +0 -306
  158. package/src/utils/__tests__/frontend-tool-inject.spec.ts +0 -350
  159. package/src/utils/__tests__/frontend-tool-integration.spec.ts +0 -199
  160. package/src/utils/__tests__/frontend-tool.utils.spec.ts +0 -272
  161. package/src/utils/__tests__/human-in-the-loop.utils.spec.ts +0 -365
  162. package/src/utils/agent-context.utils.ts +0 -133
  163. package/src/utils/agent.utils.ts +0 -239
  164. package/src/utils/chat-config.utils.ts +0 -221
  165. package/src/utils/copilotkit.utils.ts +0 -20
  166. package/src/utils/frontend-tool.utils.ts +0 -266
  167. package/src/utils/human-in-the-loop.utils.ts +0 -359
  168. 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,130 @@
1
+ import { Directive, Input, isDevMode, Inject, } from "@angular/core";
2
+ import { CopilotKitService } from "../core/copilotkit.service";
3
+ import { z } from "zod";
4
+ import * as i0 from "@angular/core";
5
+ import * as i1 from "../core/copilotkit.service";
6
+ export class CopilotKitFrontendToolDirective {
7
+ copilotkit;
8
+ isRegistered = false;
9
+ constructor(copilotkit) {
10
+ this.copilotkit = copilotkit;
11
+ }
12
+ name;
13
+ description;
14
+ parameters;
15
+ handler;
16
+ render;
17
+ followUp;
18
+ // Alternative: Accept a full tool object
19
+ tool;
20
+ ngOnInit() {
21
+ this.registerTool();
22
+ }
23
+ ngOnChanges(_changes) {
24
+ if (this.isRegistered) {
25
+ // Re-register the tool if any properties change
26
+ this.unregisterTool();
27
+ this.registerTool();
28
+ }
29
+ }
30
+ ngOnDestroy() {
31
+ this.unregisterTool();
32
+ }
33
+ registerTool() {
34
+ const tool = this.getTool();
35
+ if (!tool.name) {
36
+ if (isDevMode()) {
37
+ console.warn('CopilotKitFrontendToolDirective: "name" is required. ' +
38
+ 'Please provide a name via [name]="toolName" or ' +
39
+ "[copilotkitFrontendTool]=\"{ name: 'toolName', ... }\"");
40
+ }
41
+ return; // Don't register if no name
42
+ }
43
+ // Register the tool with CopilotKit
44
+ this.copilotkit.copilotkit.addTool(tool);
45
+ // Register the render if provided
46
+ if (tool.render) {
47
+ const currentRenders = this.copilotkit.currentRenderToolCalls();
48
+ const renderEntry = {
49
+ name: tool.name,
50
+ args: tool.parameters || z.object({}),
51
+ render: tool.render,
52
+ };
53
+ // Check for duplicate
54
+ const existingIndex = currentRenders.findIndex((r) => r.name === tool.name);
55
+ if (existingIndex !== -1) {
56
+ if (isDevMode()) {
57
+ console.warn(`[CopilotKit] Tool "${tool.name}" already has a render. ` +
58
+ `The previous render will be replaced. ` +
59
+ `This may indicate a duplicate tool registration.`);
60
+ }
61
+ const updated = [...currentRenders];
62
+ updated[existingIndex] = renderEntry;
63
+ this.copilotkit.setCurrentRenderToolCalls(updated);
64
+ }
65
+ else {
66
+ this.copilotkit.setCurrentRenderToolCalls([...currentRenders, renderEntry]);
67
+ }
68
+ }
69
+ this.isRegistered = true;
70
+ }
71
+ unregisterTool() {
72
+ if (!this.isRegistered)
73
+ return;
74
+ const tool = this.getTool();
75
+ if (tool.name) {
76
+ // Remove the tool
77
+ this.copilotkit.copilotkit.removeTool(tool.name);
78
+ // Remove the render if it exists
79
+ const currentRenders = this.copilotkit.currentRenderToolCalls();
80
+ const filtered = currentRenders.filter((r) => r.name !== tool.name);
81
+ if (filtered.length !== currentRenders.length) {
82
+ this.copilotkit.setCurrentRenderToolCalls(filtered);
83
+ }
84
+ }
85
+ this.isRegistered = false;
86
+ }
87
+ getTool() {
88
+ // If full tool object is provided, use it
89
+ if (this.tool) {
90
+ return this.tool;
91
+ }
92
+ // Otherwise, construct from individual inputs
93
+ return {
94
+ name: this.name,
95
+ description: this.description,
96
+ parameters: this.parameters,
97
+ handler: this.handler,
98
+ render: this.render,
99
+ followUp: this.followUp,
100
+ };
101
+ }
102
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CopilotKitFrontendToolDirective, deps: [{ token: CopilotKitService }], target: i0.ɵɵFactoryTarget.Directive });
103
+ 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 });
104
+ }
105
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CopilotKitFrontendToolDirective, decorators: [{
106
+ type: Directive,
107
+ args: [{
108
+ selector: "[copilotkitFrontendTool]",
109
+ standalone: true,
110
+ }]
111
+ }], ctorParameters: () => [{ type: i1.CopilotKitService, decorators: [{
112
+ type: Inject,
113
+ args: [CopilotKitService]
114
+ }] }], propDecorators: { name: [{
115
+ type: Input
116
+ }], description: [{
117
+ type: Input
118
+ }], parameters: [{
119
+ type: Input
120
+ }], handler: [{
121
+ type: Input
122
+ }], render: [{
123
+ type: Input
124
+ }], followUp: [{
125
+ type: Input
126
+ }], tool: [{
127
+ type: Input,
128
+ args: ["copilotkitFrontendTool"]
129
+ }] } });
130
+ //# 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 { 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        args: tool.parameters || z.object({}),\n        render: tool.render,\n      };\n\n      // Check for duplicate\n      const existingIndex = currentRenders.findIndex((r: ToolCallRender<unknown>) => 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<unknown>) => 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,266 @@
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
+ args: this.parameters,
171
+ render: this.createEnhancedRender(),
172
+ });
173
+ }
174
+ unregisterTool() {
175
+ if (this.toolId) {
176
+ this.copilotkit.copilotkit.removeTool(this.toolId);
177
+ this.copilotkit.unregisterToolRender(this.name);
178
+ this.toolId = undefined;
179
+ }
180
+ }
181
+ createEnhancedRender() {
182
+ // If it's a template, we need to wrap it with our respond function
183
+ // This is handled by returning a special marker that the render component
184
+ // will recognize and enhance with the respond function
185
+ // Store reference to this directive instance for the render component
186
+ this.render.__humanInTheLoopDirective = this;
187
+ this.render.__humanInTheLoopStatus = this.statusSignal;
188
+ return this.render;
189
+ }
190
+ handleResponse(result) {
191
+ if (this.resolvePromise) {
192
+ this.resolvePromise(result);
193
+ this.updateStatus(ToolCallStatus.Complete);
194
+ this.resolvePromise = null;
195
+ this.responseProvided.emit(result);
196
+ this.executionCompleted.emit(result);
197
+ }
198
+ }
199
+ updateStatus(status) {
200
+ this._status = status;
201
+ this.statusSignal.set(status);
202
+ this.statusChange.emit(status);
203
+ }
204
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CopilotKitHumanInTheLoopDirective, deps: [{ token: CopilotKitService }], target: i0.ɵɵFactoryTarget.Directive });
205
+ 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 });
206
+ }
207
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CopilotKitHumanInTheLoopDirective, decorators: [{
208
+ type: Directive,
209
+ args: [{
210
+ selector: "[copilotkitHumanInTheLoop]",
211
+ standalone: true,
212
+ }]
213
+ }], ctorParameters: () => [{ type: i1.CopilotKitService, decorators: [{
214
+ type: Inject,
215
+ args: [CopilotKitService]
216
+ }] }], propDecorators: { name: [{
217
+ type: Input
218
+ }], description: [{
219
+ type: Input
220
+ }], parameters: [{
221
+ type: Input
222
+ }], render: [{
223
+ type: Input
224
+ }], enabled: [{
225
+ type: Input
226
+ }], config: [{
227
+ type: Input,
228
+ args: ["copilotkitHumanInTheLoop"]
229
+ }], statusChange: [{
230
+ type: Output
231
+ }], status: [{
232
+ type: Input
233
+ }], responseProvided: [{
234
+ type: Output
235
+ }], executionStarted: [{
236
+ type: Output
237
+ }], executionCompleted: [{
238
+ type: Output
239
+ }] } });
240
+ /**
241
+ * Helper directive to provide respond function in templates.
242
+ * This would be used internally by the tool render component.
243
+ *
244
+ * @internal
245
+ */
246
+ export class CopilotKitHumanInTheLoopRespondDirective {
247
+ copilotkitHumanInTheLoopRespond;
248
+ /**
249
+ * Convenience method for templates to call respond.
250
+ */
251
+ respond(result) {
252
+ this.copilotkitHumanInTheLoopRespond?.(result);
253
+ }
254
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CopilotKitHumanInTheLoopRespondDirective, deps: [], target: i0.ɵɵFactoryTarget.Directive });
255
+ static ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "18.2.13", type: CopilotKitHumanInTheLoopRespondDirective, isStandalone: true, selector: "[copilotkitHumanInTheLoopRespond]", inputs: { copilotkitHumanInTheLoopRespond: "copilotkitHumanInTheLoopRespond" }, ngImport: i0 });
256
+ }
257
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CopilotKitHumanInTheLoopRespondDirective, decorators: [{
258
+ type: Directive,
259
+ args: [{
260
+ selector: "[copilotkitHumanInTheLoopRespond]",
261
+ standalone: true,
262
+ }]
263
+ }], propDecorators: { copilotkitHumanInTheLoopRespond: [{
264
+ type: Input
265
+ }] } });
266
+ //# 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      args: this.parameters,\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"]}