@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.
- package/README.md +3 -3
- package/dist/README.md +3 -3
- package/dist/components/chat/copilot-chat-assistant-message.component.d.ts +10 -10
- package/dist/components/chat/copilot-chat-message-view.component.d.ts +42 -42
- package/dist/components/chat/copilot-chat-view.component.d.ts +14 -14
- package/dist/esm2022/components/chat/copilot-chat-assistant-message-buttons.component.mjs +384 -0
- package/dist/esm2022/components/chat/copilot-chat-assistant-message-renderer.component.mjs +286 -0
- package/dist/esm2022/components/chat/copilot-chat-assistant-message-toolbar.component.mjs +27 -0
- package/dist/esm2022/components/chat/copilot-chat-assistant-message.component.mjs +433 -0
- package/dist/esm2022/components/chat/copilot-chat-assistant-message.types.mjs +2 -0
- package/dist/esm2022/components/chat/copilot-chat-audio-recorder.component.mjs +202 -0
- package/dist/esm2022/components/chat/copilot-chat-buttons.component.mjs +321 -0
- package/dist/esm2022/components/chat/copilot-chat-input-defaults.mjs +38 -0
- package/dist/esm2022/components/chat/copilot-chat-input.component.mjs +666 -0
- package/dist/esm2022/components/chat/copilot-chat-input.types.mjs +10 -0
- package/dist/esm2022/components/chat/copilot-chat-message-view-cursor.component.mjs +45 -0
- package/dist/esm2022/components/chat/copilot-chat-message-view.component.mjs +296 -0
- package/dist/esm2022/components/chat/copilot-chat-message-view.types.mjs +2 -0
- package/dist/esm2022/components/chat/copilot-chat-textarea.component.mjs +188 -0
- package/dist/esm2022/components/chat/copilot-chat-tool-calls-view.component.mjs +216 -0
- package/dist/esm2022/components/chat/copilot-chat-toolbar.component.mjs +25 -0
- package/dist/esm2022/components/chat/copilot-chat-tools-menu.component.mjs +199 -0
- package/dist/esm2022/components/chat/copilot-chat-user-message-branch-navigation.component.mjs +137 -0
- package/dist/esm2022/components/chat/copilot-chat-user-message-buttons.component.mjs +207 -0
- package/dist/esm2022/components/chat/copilot-chat-user-message-renderer.component.mjs +35 -0
- package/dist/esm2022/components/chat/copilot-chat-user-message-toolbar.component.mjs +34 -0
- package/dist/esm2022/components/chat/copilot-chat-user-message.component.mjs +341 -0
- package/dist/esm2022/components/chat/copilot-chat-user-message.types.mjs +2 -0
- package/dist/esm2022/components/chat/copilot-chat-view-disclaimer.component.mjs +52 -0
- package/dist/esm2022/components/chat/copilot-chat-view-feather.component.mjs +55 -0
- package/dist/esm2022/components/chat/copilot-chat-view-handlers.service.mjs +19 -0
- package/dist/esm2022/components/chat/copilot-chat-view-input-container.component.mjs +110 -0
- package/dist/esm2022/components/chat/copilot-chat-view-scroll-to-bottom-button.component.mjs +93 -0
- package/dist/esm2022/components/chat/copilot-chat-view-scroll-view.component.mjs +443 -0
- package/dist/esm2022/components/chat/copilot-chat-view.component.mjs +479 -0
- package/dist/esm2022/components/chat/copilot-chat-view.types.mjs +2 -0
- package/dist/esm2022/components/chat/copilot-chat.component.mjs +214 -0
- package/dist/esm2022/components/copilotkit-tool-render.component.mjs +153 -0
- package/dist/esm2022/copilotkitnext-angular.mjs +5 -0
- package/dist/esm2022/core/chat-configuration/chat-configuration.providers.mjs +65 -0
- package/dist/esm2022/core/chat-configuration/chat-configuration.service.mjs +145 -0
- package/dist/esm2022/core/chat-configuration/chat-configuration.types.mjs +26 -0
- package/dist/esm2022/core/copilotkit.providers.mjs +34 -0
- package/dist/esm2022/core/copilotkit.service.mjs +430 -0
- package/dist/esm2022/core/copilotkit.types.mjs +12 -0
- package/dist/esm2022/directives/copilotkit-agent-context.directive.mjs +130 -0
- package/dist/esm2022/directives/copilotkit-agent.directive.mjs +217 -0
- package/dist/esm2022/directives/copilotkit-chat-config.directive.mjs +218 -0
- package/dist/esm2022/directives/copilotkit-config.directive.mjs +94 -0
- package/dist/esm2022/directives/copilotkit-frontend-tool.directive.mjs +130 -0
- package/dist/esm2022/directives/copilotkit-human-in-the-loop.directive.mjs +266 -0
- package/dist/esm2022/directives/stick-to-bottom.directive.mjs +181 -0
- package/dist/esm2022/index.mjs +70 -0
- package/dist/esm2022/lib/directives/tooltip.directive.mjs +211 -0
- package/dist/esm2022/lib/slots/copilot-slot.component.mjs +144 -0
- package/dist/esm2022/lib/slots/slot.types.mjs +6 -0
- package/dist/esm2022/lib/slots/slot.utils.mjs +222 -0
- package/dist/esm2022/lib/utils.mjs +10 -0
- package/dist/esm2022/services/resize-observer.service.mjs +152 -0
- package/dist/esm2022/services/scroll-position.service.mjs +124 -0
- package/dist/esm2022/types/frontend-tool.mjs +2 -0
- package/dist/esm2022/types/human-in-the-loop.mjs +2 -0
- package/dist/esm2022/utils/agent-context.utils.mjs +114 -0
- package/dist/esm2022/utils/agent.utils.mjs +204 -0
- package/dist/esm2022/utils/chat-config.utils.mjs +186 -0
- package/dist/esm2022/utils/copilotkit.utils.mjs +20 -0
- package/dist/esm2022/utils/frontend-tool.utils.mjs +228 -0
- package/dist/esm2022/utils/human-in-the-loop.utils.mjs +296 -0
- package/dist/fesm2022/copilotkitnext-angular.mjs +163 -164
- package/dist/fesm2022/copilotkitnext-angular.mjs.map +1 -1
- package/package.json +21 -18
- package/vitest.config.mts +32 -21
- package/.turbo/turbo-build.log +0 -38
- package/.turbo/turbo-check-types.log +0 -0
- package/.turbo/turbo-test.log +0 -71
- package/ng-package.json +0 -19
- package/src/components/chat/__tests__/copilot-chat-assistant-message.component.spec.ts +0 -282
- package/src/components/chat/__tests__/copilot-chat-input.component.spec.ts +0 -419
- package/src/components/chat/__tests__/copilot-chat-message-view.component.spec.ts +0 -372
- package/src/components/chat/__tests__/copilot-chat-user-message.component.spec.ts +0 -249
- package/src/components/chat/copilot-chat-assistant-message-buttons.component.ts +0 -292
- package/src/components/chat/copilot-chat-assistant-message-renderer.component.ts +0 -472
- package/src/components/chat/copilot-chat-assistant-message-toolbar.component.ts +0 -29
- package/src/components/chat/copilot-chat-assistant-message.component.ts +0 -463
- package/src/components/chat/copilot-chat-assistant-message.types.ts +0 -50
- package/src/components/chat/copilot-chat-audio-recorder.component.ts +0 -241
- package/src/components/chat/copilot-chat-buttons.component.ts +0 -308
- package/src/components/chat/copilot-chat-buttons.component.ts.bak +0 -471
- package/src/components/chat/copilot-chat-input-defaults.ts +0 -47
- package/src/components/chat/copilot-chat-input.component.ts +0 -512
- package/src/components/chat/copilot-chat-input.types.ts +0 -148
- package/src/components/chat/copilot-chat-message-view-cursor.component.ts +0 -51
- package/src/components/chat/copilot-chat-message-view.component.ts +0 -233
- package/src/components/chat/copilot-chat-message-view.types.ts +0 -39
- package/src/components/chat/copilot-chat-textarea.component.ts +0 -220
- package/src/components/chat/copilot-chat-tool-calls-view.component.ts +0 -261
- package/src/components/chat/copilot-chat-toolbar.component.ts +0 -35
- package/src/components/chat/copilot-chat-tools-menu.component.ts +0 -185
- package/src/components/chat/copilot-chat-user-message-branch-navigation.component.ts +0 -121
- package/src/components/chat/copilot-chat-user-message-buttons.component.ts +0 -170
- package/src/components/chat/copilot-chat-user-message-renderer.component.ts +0 -37
- package/src/components/chat/copilot-chat-user-message-toolbar.component.ts +0 -37
- package/src/components/chat/copilot-chat-user-message.component.ts +0 -247
- package/src/components/chat/copilot-chat-user-message.types.ts +0 -42
- package/src/components/chat/copilot-chat-view-disclaimer.component.ts +0 -51
- package/src/components/chat/copilot-chat-view-feather.component.ts +0 -47
- package/src/components/chat/copilot-chat-view-handlers.service.ts +0 -14
- package/src/components/chat/copilot-chat-view-input-container.component.ts +0 -87
- package/src/components/chat/copilot-chat-view-scroll-to-bottom-button.component.ts +0 -79
- package/src/components/chat/copilot-chat-view-scroll-view.component.ts +0 -322
- package/src/components/chat/copilot-chat-view.component.ts +0 -420
- package/src/components/chat/copilot-chat-view.types.ts +0 -52
- package/src/components/chat/copilot-chat.component.ts +0 -232
- package/src/components/copilotkit-tool-render.component.ts +0 -169
- package/src/core/__tests__/copilotkit.service.spec.ts +0 -1051
- package/src/core/__tests__/copilotkit.service.wildcard.spec.ts +0 -316
- package/src/core/chat-configuration/__tests__/chat-configuration.service.spec.ts +0 -287
- package/src/core/chat-configuration/chat-configuration.providers.ts +0 -71
- package/src/core/chat-configuration/chat-configuration.service.ts +0 -162
- package/src/core/chat-configuration/chat-configuration.types.ts +0 -57
- package/src/core/copilotkit.providers.ts +0 -59
- package/src/core/copilotkit.service.ts +0 -542
- package/src/core/copilotkit.types.ts +0 -132
- package/src/directives/__tests__/copilotkit-agent-context.directive.spec.ts +0 -384
- package/src/directives/__tests__/copilotkit-agent.directive.spec.ts +0 -253
- package/src/directives/__tests__/copilotkit-chat-config.directive.spec.ts +0 -385
- package/src/directives/__tests__/copilotkit-config.directive.spec.ts +0 -69
- package/src/directives/__tests__/copilotkit-frontend-tool-simple.directive.spec.ts +0 -60
- package/src/directives/__tests__/copilotkit-frontend-tool.directive.spec.ts +0 -108
- package/src/directives/__tests__/copilotkit-human-in-the-loop.directive.spec.ts +0 -452
- package/src/directives/copilotkit-agent-context.directive.ts +0 -138
- package/src/directives/copilotkit-agent.directive.ts +0 -225
- package/src/directives/copilotkit-chat-config.directive.ts +0 -241
- package/src/directives/copilotkit-config.directive.ts +0 -81
- package/src/directives/copilotkit-frontend-tool.directive.ts +0 -145
- package/src/directives/copilotkit-human-in-the-loop.directive.ts +0 -281
- package/src/directives/stick-to-bottom.directive.ts +0 -204
- package/src/index.ts +0 -105
- package/src/lib/directives/tooltip.directive.ts +0 -292
- package/src/lib/slots/__tests__/slot.utils.spec.ts +0 -377
- package/src/lib/slots/copilot-slot.component.ts +0 -135
- package/src/lib/slots/index.ts +0 -3
- package/src/lib/slots/slot.types.ts +0 -64
- package/src/lib/slots/slot.utils.ts +0 -289
- package/src/lib/utils.ts +0 -10
- package/src/public-api.ts +0 -1
- package/src/services/resize-observer.service.ts +0 -181
- package/src/services/scroll-position.service.ts +0 -169
- package/src/styles/globals.css +0 -266
- package/src/styles/index.css +0 -3
- package/src/test-setup.ts +0 -15
- package/src/testing/index.ts +0 -3
- package/src/testing/testing.utils.ts +0 -248
- package/src/types/frontend-tool.ts +0 -44
- package/src/types/human-in-the-loop.ts +0 -52
- package/src/utils/__tests__/agent.utils.spec.ts +0 -234
- package/src/utils/__tests__/chat-config.utils.spec.ts +0 -306
- package/src/utils/__tests__/frontend-tool-inject.spec.ts +0 -350
- package/src/utils/__tests__/frontend-tool-integration.spec.ts +0 -199
- package/src/utils/__tests__/frontend-tool.utils.spec.ts +0 -272
- package/src/utils/__tests__/human-in-the-loop.utils.spec.ts +0 -365
- package/src/utils/agent-context.utils.ts +0 -133
- package/src/utils/agent.utils.ts +0 -239
- package/src/utils/chat-config.utils.ts +0 -221
- package/src/utils/copilotkit.utils.ts +0 -20
- package/src/utils/frontend-tool.utils.ts +0 -266
- package/src/utils/human-in-the-loop.utils.ts +0 -359
- package/tsconfig.spec.json +0 -12
|
@@ -0,0 +1,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"]}
|