@copilotkitnext/angular 0.0.0-max-changeset-20260109174803
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/LICENSE +10 -0
- package/README.md +530 -0
- package/dist/LICENSE +10 -0
- package/dist/README.md +530 -0
- package/dist/esm2022/copilotkitnext-angular.mjs +5 -0
- package/dist/esm2022/index.mjs +2 -0
- package/dist/esm2022/lib/agent-context.mjs +25 -0
- package/dist/esm2022/lib/agent.mjs +73 -0
- package/dist/esm2022/lib/chat-config.mjs +35 -0
- package/dist/esm2022/lib/chat-state.mjs +18 -0
- package/dist/esm2022/lib/components/chat/copilot-chat-assistant-message-buttons.mjs +344 -0
- package/dist/esm2022/lib/components/chat/copilot-chat-assistant-message-renderer.mjs +260 -0
- package/dist/esm2022/lib/components/chat/copilot-chat-assistant-message-toolbar.mjs +22 -0
- package/dist/esm2022/lib/components/chat/copilot-chat-assistant-message.mjs +415 -0
- package/dist/esm2022/lib/components/chat/copilot-chat-assistant-message.types.mjs +2 -0
- package/dist/esm2022/lib/components/chat/copilot-chat-audio-recorder.mjs +196 -0
- package/dist/esm2022/lib/components/chat/copilot-chat-buttons.mjs +299 -0
- package/dist/esm2022/lib/components/chat/copilot-chat-input-defaults.mjs +39 -0
- package/dist/esm2022/lib/components/chat/copilot-chat-input.mjs +634 -0
- package/dist/esm2022/lib/components/chat/copilot-chat-input.types.mjs +10 -0
- package/dist/esm2022/lib/components/chat/copilot-chat-message-view-cursor.mjs +27 -0
- package/dist/esm2022/lib/components/chat/copilot-chat-message-view.mjs +268 -0
- package/dist/esm2022/lib/components/chat/copilot-chat-message-view.types.mjs +2 -0
- package/dist/esm2022/lib/components/chat/copilot-chat-textarea.mjs +139 -0
- package/dist/esm2022/lib/components/chat/copilot-chat-tool-calls-view.mjs +36 -0
- package/dist/esm2022/lib/components/chat/copilot-chat-toolbar.mjs +20 -0
- package/dist/esm2022/lib/components/chat/copilot-chat-tools-menu.mjs +203 -0
- package/dist/esm2022/lib/components/chat/copilot-chat-user-message-branch-navigation.mjs +98 -0
- package/dist/esm2022/lib/components/chat/copilot-chat-user-message-buttons.mjs +182 -0
- package/dist/esm2022/lib/components/chat/copilot-chat-user-message-renderer.mjs +28 -0
- package/dist/esm2022/lib/components/chat/copilot-chat-user-message-toolbar.mjs +25 -0
- package/dist/esm2022/lib/components/chat/copilot-chat-user-message.mjs +308 -0
- package/dist/esm2022/lib/components/chat/copilot-chat-user-message.types.mjs +2 -0
- package/dist/esm2022/lib/components/chat/copilot-chat-view-disclaimer.mjs +48 -0
- package/dist/esm2022/lib/components/chat/copilot-chat-view-feather.mjs +41 -0
- package/dist/esm2022/lib/components/chat/copilot-chat-view-handlers.mjs +19 -0
- package/dist/esm2022/lib/components/chat/copilot-chat-view-input-container.mjs +96 -0
- package/dist/esm2022/lib/components/chat/copilot-chat-view-scroll-to-bottom-button.mjs +89 -0
- package/dist/esm2022/lib/components/chat/copilot-chat-view-scroll-view.mjs +456 -0
- package/dist/esm2022/lib/components/chat/copilot-chat-view.mjs +404 -0
- package/dist/esm2022/lib/components/chat/copilot-chat-view.types.mjs +2 -0
- package/dist/esm2022/lib/components/chat/copilot-chat.mjs +167 -0
- package/dist/esm2022/lib/config.mjs +9 -0
- package/dist/esm2022/lib/copilotkit.mjs +115 -0
- package/dist/esm2022/lib/directives/copilotkit-agent-context.mjs +130 -0
- package/dist/esm2022/lib/directives/stick-to-bottom.mjs +170 -0
- package/dist/esm2022/lib/directives/tooltip.mjs +217 -0
- package/dist/esm2022/lib/human-in-the-loop.mjs +19 -0
- package/dist/esm2022/lib/render-tool-calls.mjs +138 -0
- package/dist/esm2022/lib/resize-observer.mjs +152 -0
- package/dist/esm2022/lib/scroll-position.mjs +124 -0
- package/dist/esm2022/lib/slots/copilot-slot.mjs +156 -0
- package/dist/esm2022/lib/slots/index.mjs +4 -0
- package/dist/esm2022/lib/slots/slot.types.mjs +6 -0
- package/dist/esm2022/lib/slots/slot.utils.mjs +235 -0
- package/dist/esm2022/lib/tools.mjs +31 -0
- package/dist/esm2022/lib/utils.mjs +10 -0
- package/dist/esm2022/public-api.mjs +48 -0
- package/dist/fesm2022/copilotkitnext-angular.mjs +6310 -0
- package/dist/fesm2022/copilotkitnext-angular.mjs.map +1 -0
- package/dist/index.d.ts +1 -0
- package/dist/lib/agent-context.d.ts +12 -0
- package/dist/lib/agent.d.ts +68 -0
- package/dist/lib/chat-config.d.ts +23 -0
- package/dist/lib/chat-state.d.ts +10 -0
- package/dist/lib/components/chat/copilot-chat-assistant-message-buttons.d.ts +68 -0
- package/dist/lib/components/chat/copilot-chat-assistant-message-renderer.d.ts +26 -0
- package/dist/lib/components/chat/copilot-chat-assistant-message-toolbar.d.ts +7 -0
- package/dist/lib/components/chat/copilot-chat-assistant-message.d.ts +208 -0
- package/dist/lib/components/chat/copilot-chat-assistant-message.types.d.ts +31 -0
- package/dist/lib/components/chat/copilot-chat-audio-recorder.d.ts +40 -0
- package/dist/lib/components/chat/copilot-chat-buttons.d.ts +65 -0
- package/dist/lib/components/chat/copilot-chat-input-defaults.d.ts +38 -0
- package/dist/lib/components/chat/copilot-chat-input.d.ts +133 -0
- package/dist/lib/components/chat/copilot-chat-input.types.d.ts +129 -0
- package/dist/lib/components/chat/copilot-chat-message-view-cursor.d.ts +11 -0
- package/dist/lib/components/chat/copilot-chat-message-view.d.ts +430 -0
- package/dist/lib/components/chat/copilot-chat-message-view.types.d.ts +24 -0
- package/dist/lib/components/chat/copilot-chat-textarea.d.ts +41 -0
- package/dist/lib/components/chat/copilot-chat-tool-calls-view.d.ts +70 -0
- package/dist/lib/components/chat/copilot-chat-toolbar.d.ts +7 -0
- package/dist/lib/components/chat/copilot-chat-tools-menu.d.ts +20 -0
- package/dist/lib/components/chat/copilot-chat-user-message-branch-navigation.d.ts +35 -0
- package/dist/lib/components/chat/copilot-chat-user-message-buttons.d.ts +35 -0
- package/dist/lib/components/chat/copilot-chat-user-message-renderer.d.ts +8 -0
- package/dist/lib/components/chat/copilot-chat-user-message-toolbar.d.ts +7 -0
- package/dist/lib/components/chat/copilot-chat-user-message.d.ts +71 -0
- package/dist/lib/components/chat/copilot-chat-user-message.types.d.ts +27 -0
- package/dist/lib/components/chat/copilot-chat-view-disclaimer.d.ts +15 -0
- package/dist/lib/components/chat/copilot-chat-view-feather.d.ts +15 -0
- package/dist/lib/components/chat/copilot-chat-view-handlers.d.ts +11 -0
- package/dist/lib/components/chat/copilot-chat-view-input-container.d.ts +23 -0
- package/dist/lib/components/chat/copilot-chat-view-scroll-to-bottom-button.d.ts +16 -0
- package/dist/lib/components/chat/copilot-chat-view-scroll-view.d.ts +129 -0
- package/dist/lib/components/chat/copilot-chat-view.d.ts +284 -0
- package/dist/lib/components/chat/copilot-chat-view.types.d.ts +42 -0
- package/dist/lib/components/chat/copilot-chat.d.ts +82 -0
- package/dist/lib/config.d.ts +16 -0
- package/dist/lib/copilotkit.d.ts +29 -0
- package/dist/lib/directives/copilotkit-agent-context.d.ts +68 -0
- package/dist/lib/directives/stick-to-bottom.d.ts +62 -0
- package/dist/lib/directives/tooltip.d.ts +33 -0
- package/dist/lib/human-in-the-loop.d.ts +13 -0
- package/dist/lib/render-tool-calls.d.ts +90 -0
- package/dist/lib/resize-observer.d.ts +44 -0
- package/dist/lib/scroll-position.d.ts +50 -0
- package/dist/lib/slots/copilot-slot.d.ts +34 -0
- package/dist/lib/slots/index.d.ts +3 -0
- package/dist/lib/slots/slot.types.d.ts +55 -0
- package/dist/lib/slots/slot.utils.d.ts +110 -0
- package/dist/lib/tools.d.ts +65 -0
- package/dist/lib/utils.d.ts +6 -0
- package/dist/public-api.d.ts +47 -0
- package/dist/styles.css +1882 -0
- package/eslint.config.mjs +20 -0
- package/package.json +101 -0
- package/tsconfig.json +33 -0
- package/vitest.config.mts +45 -0
|
@@ -0,0 +1,167 @@
|
|
|
1
|
+
import { Component, input, ChangeDetectionStrategy, ViewEncapsulation, signal, effect, ChangeDetectorRef, Injector, computed, inject, } from "@angular/core";
|
|
2
|
+
import { CommonModule } from "@angular/common";
|
|
3
|
+
import { CopilotChatView } from "./copilot-chat-view";
|
|
4
|
+
import { DEFAULT_AGENT_ID, randomUUID } from "@copilotkitnext/shared";
|
|
5
|
+
import { injectAgentStore } from "../../agent";
|
|
6
|
+
import { ChatState } from "../../chat-state";
|
|
7
|
+
import * as i0 from "@angular/core";
|
|
8
|
+
/**
|
|
9
|
+
* CopilotChat component - Angular equivalent of React's <CopilotChat>
|
|
10
|
+
* Provides a complete chat interface that wires an agent to the chat view
|
|
11
|
+
*
|
|
12
|
+
* @example
|
|
13
|
+
* ```html
|
|
14
|
+
* <copilot-chat [agentId]="'default'" [threadId]="'abc123'"></copilot-chat>
|
|
15
|
+
* ```
|
|
16
|
+
*/
|
|
17
|
+
export class CopilotChat {
|
|
18
|
+
inputValue = signal("");
|
|
19
|
+
agentId = input();
|
|
20
|
+
threadId = input();
|
|
21
|
+
inputComponent = input();
|
|
22
|
+
resolvedAgentId = computed(() => this.agentId() ?? DEFAULT_AGENT_ID);
|
|
23
|
+
agentStore = injectAgentStore(this.resolvedAgentId);
|
|
24
|
+
// readonly chatConfig = injectChatConfig();
|
|
25
|
+
cdr = inject(ChangeDetectorRef);
|
|
26
|
+
injector = inject(Injector);
|
|
27
|
+
messages = computed(() => this.agentStore()?.messages());
|
|
28
|
+
isRunning = computed(() => this.agentStore()?.isRunning());
|
|
29
|
+
showCursor = signal(false);
|
|
30
|
+
generatedThreadId = randomUUID();
|
|
31
|
+
hasConnectedOnce = false;
|
|
32
|
+
constructor() {
|
|
33
|
+
// Connect once when agent becomes available
|
|
34
|
+
// Connect once when agent becomes available
|
|
35
|
+
effect(() => {
|
|
36
|
+
const a = this.agentStore()?.agent;
|
|
37
|
+
if (!a)
|
|
38
|
+
return;
|
|
39
|
+
// Apply thread id when agent is available
|
|
40
|
+
a.threadId = this.threadId() || this.generatedThreadId;
|
|
41
|
+
if (!this.hasConnectedOnce) {
|
|
42
|
+
this.hasConnectedOnce = true;
|
|
43
|
+
if ("isCopilotKitAgent" in a) {
|
|
44
|
+
this.connectToAgent(a);
|
|
45
|
+
}
|
|
46
|
+
else {
|
|
47
|
+
// Non-CopilotKit agent: nothing to connect; keep default cursor state
|
|
48
|
+
}
|
|
49
|
+
}
|
|
50
|
+
}, { allowSignalWrites: true });
|
|
51
|
+
// Keep agent threadId in sync with input
|
|
52
|
+
effect(() => {
|
|
53
|
+
const a = this.agentStore()?.agent;
|
|
54
|
+
if (a) {
|
|
55
|
+
a.threadId = this.threadId() || this.generatedThreadId;
|
|
56
|
+
}
|
|
57
|
+
});
|
|
58
|
+
}
|
|
59
|
+
async connectToAgent(agent) {
|
|
60
|
+
if (!agent)
|
|
61
|
+
return;
|
|
62
|
+
this.showCursor.set(true);
|
|
63
|
+
this.cdr.markForCheck();
|
|
64
|
+
try {
|
|
65
|
+
await agent.runAgent({ forwardedProps: { __copilotkitConnect: true } }, {
|
|
66
|
+
onTextMessageStartEvent: () => {
|
|
67
|
+
this.showCursor.set(false);
|
|
68
|
+
this.cdr.detectChanges();
|
|
69
|
+
},
|
|
70
|
+
onToolCallStartEvent: () => {
|
|
71
|
+
this.showCursor.set(false);
|
|
72
|
+
this.cdr.detectChanges();
|
|
73
|
+
},
|
|
74
|
+
});
|
|
75
|
+
this.showCursor.set(false);
|
|
76
|
+
this.cdr.markForCheck();
|
|
77
|
+
}
|
|
78
|
+
catch (error) {
|
|
79
|
+
console.error("Failed to connect to agent:", error);
|
|
80
|
+
this.showCursor.set(false);
|
|
81
|
+
this.cdr.markForCheck();
|
|
82
|
+
}
|
|
83
|
+
}
|
|
84
|
+
async submitInput(value) {
|
|
85
|
+
const agent = this.agentStore()?.agent;
|
|
86
|
+
if (!agent || !value.trim())
|
|
87
|
+
return;
|
|
88
|
+
// Add user message
|
|
89
|
+
const userMessage = {
|
|
90
|
+
id: randomUUID(),
|
|
91
|
+
role: "user",
|
|
92
|
+
content: value,
|
|
93
|
+
};
|
|
94
|
+
agent.addMessage(userMessage);
|
|
95
|
+
// Clear the input
|
|
96
|
+
this.inputValue.set("");
|
|
97
|
+
// Show cursor while processing
|
|
98
|
+
this.showCursor.set(true);
|
|
99
|
+
this.cdr.markForCheck();
|
|
100
|
+
// Run the agent with named subscriber callbacks
|
|
101
|
+
try {
|
|
102
|
+
await agent.runAgent({}, {
|
|
103
|
+
onTextMessageStartEvent: () => {
|
|
104
|
+
this.showCursor.set(false);
|
|
105
|
+
this.cdr.detectChanges();
|
|
106
|
+
},
|
|
107
|
+
onToolCallStartEvent: () => {
|
|
108
|
+
this.showCursor.set(false);
|
|
109
|
+
this.cdr.detectChanges();
|
|
110
|
+
},
|
|
111
|
+
});
|
|
112
|
+
}
|
|
113
|
+
catch (error) {
|
|
114
|
+
console.error("Agent run error:", error);
|
|
115
|
+
}
|
|
116
|
+
finally {
|
|
117
|
+
this.showCursor.set(false);
|
|
118
|
+
this.cdr.markForCheck();
|
|
119
|
+
}
|
|
120
|
+
}
|
|
121
|
+
changeInput(value) {
|
|
122
|
+
this.inputValue.set(value);
|
|
123
|
+
}
|
|
124
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: CopilotChat, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
125
|
+
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.1.0", version: "18.2.14", type: CopilotChat, isStandalone: true, selector: "copilot-chat", inputs: { agentId: { classPropertyName: "agentId", publicName: "agentId", isSignal: true, isRequired: false, transformFunction: null }, threadId: { classPropertyName: "threadId", publicName: "threadId", isSignal: true, isRequired: false, transformFunction: null }, inputComponent: { classPropertyName: "inputComponent", publicName: "inputComponent", isSignal: true, isRequired: false, transformFunction: null } }, providers: [
|
|
126
|
+
{
|
|
127
|
+
provide: ChatState,
|
|
128
|
+
useExisting: CopilotChat,
|
|
129
|
+
},
|
|
130
|
+
], ngImport: i0, template: `
|
|
131
|
+
<copilot-chat-view
|
|
132
|
+
[messages]="messages() ?? []"
|
|
133
|
+
[autoScroll]="true"
|
|
134
|
+
[messageViewClass]="'w-full'"
|
|
135
|
+
[showCursor]="showCursor()"
|
|
136
|
+
[inputComponent]="inputComponent()"
|
|
137
|
+
>
|
|
138
|
+
</copilot-chat-view>
|
|
139
|
+
`, isInline: true, dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "component", type: CopilotChatView, selector: "copilot-chat-view", inputs: ["messages", "autoScroll", "showCursor", "messageViewComponent", "messageViewTemplate", "messageViewClass", "scrollViewComponent", "scrollViewTemplate", "scrollViewClass", "scrollToBottomButtonComponent", "scrollToBottomButtonTemplate", "scrollToBottomButtonClass", "inputComponent", "inputTemplate", "inputContainerComponent", "inputContainerTemplate", "inputContainerClass", "featherComponent", "featherTemplate", "featherClass", "disclaimerComponent", "disclaimerTemplate", "disclaimerClass", "disclaimerText"], outputs: ["assistantMessageThumbsUp", "assistantMessageThumbsDown", "assistantMessageReadAloud", "assistantMessageRegenerate", "userMessageCopy", "userMessageEdit"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None });
|
|
140
|
+
}
|
|
141
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: CopilotChat, decorators: [{
|
|
142
|
+
type: Component,
|
|
143
|
+
args: [{
|
|
144
|
+
selector: "copilot-chat",
|
|
145
|
+
standalone: true,
|
|
146
|
+
imports: [CommonModule, CopilotChatView],
|
|
147
|
+
changeDetection: ChangeDetectionStrategy.OnPush,
|
|
148
|
+
encapsulation: ViewEncapsulation.None,
|
|
149
|
+
template: `
|
|
150
|
+
<copilot-chat-view
|
|
151
|
+
[messages]="messages() ?? []"
|
|
152
|
+
[autoScroll]="true"
|
|
153
|
+
[messageViewClass]="'w-full'"
|
|
154
|
+
[showCursor]="showCursor()"
|
|
155
|
+
[inputComponent]="inputComponent()"
|
|
156
|
+
>
|
|
157
|
+
</copilot-chat-view>
|
|
158
|
+
`,
|
|
159
|
+
providers: [
|
|
160
|
+
{
|
|
161
|
+
provide: ChatState,
|
|
162
|
+
useExisting: CopilotChat,
|
|
163
|
+
},
|
|
164
|
+
],
|
|
165
|
+
}]
|
|
166
|
+
}], ctorParameters: () => [] });
|
|
167
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"copilot-chat.js","sourceRoot":"","sources":["../../../../../src/lib/components/chat/copilot-chat.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  input,\n  ChangeDetectionStrategy,\n  ViewEncapsulation,\n  signal,\n  effect,\n  ChangeDetectorRef,\n  Injector,\n  Type,\n  computed,\n  inject,\n} from \"@angular/core\";\nimport { CommonModule } from \"@angular/common\";\nimport { CopilotChatView } from \"./copilot-chat-view\";\n\nimport { DEFAULT_AGENT_ID, randomUUID } from \"@copilotkitnext/shared\";\nimport { Message, AbstractAgent } from \"@ag-ui/client\";\nimport { injectAgentStore } from \"../../agent\";\nimport { ChatState } from \"../../chat-state\";\n\n/**\n * CopilotChat component - Angular equivalent of React's <CopilotChat>\n * Provides a complete chat interface that wires an agent to the chat view\n *\n * @example\n * ```html\n * <copilot-chat [agentId]=\"'default'\" [threadId]=\"'abc123'\"></copilot-chat>\n * ```\n */\n@Component({\n  selector: \"copilot-chat\",\n  standalone: true,\n  imports: [CommonModule, CopilotChatView],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  encapsulation: ViewEncapsulation.None,\n  template: `\n    <copilot-chat-view\n      [messages]=\"messages() ?? []\"\n      [autoScroll]=\"true\"\n      [messageViewClass]=\"'w-full'\"\n      [showCursor]=\"showCursor()\"\n      [inputComponent]=\"inputComponent()\"\n    >\n    </copilot-chat-view>\n  `,\n  providers: [\n    {\n      provide: ChatState,\n      useExisting: CopilotChat,\n    },\n  ],\n})\nexport class CopilotChat implements ChatState {\n  readonly inputValue = signal<string>(\"\");\n  readonly agentId = input<string | undefined>();\n  readonly threadId = input<string | undefined>();\n  readonly inputComponent = input<Type<any> | undefined>();\n  private readonly resolvedAgentId = computed(\n    () => this.agentId() ?? DEFAULT_AGENT_ID\n  );\n  readonly agentStore = injectAgentStore(this.resolvedAgentId);\n  // readonly chatConfig = injectChatConfig();\n  readonly cdr = inject(ChangeDetectorRef);\n  readonly injector = inject(Injector);\n\n  protected messages = computed(() => this.agentStore()?.messages());\n  protected isRunning = computed(() => this.agentStore()?.isRunning());\n  protected showCursor = signal<boolean>(false);\n\n  private generatedThreadId: string = randomUUID();\n  private hasConnectedOnce = false;\n\n  constructor() {\n    // Connect once when agent becomes available\n    // Connect once when agent becomes available\n    effect(\n      () => {\n        const a = this.agentStore()?.agent;\n        if (!a) return;\n        // Apply thread id when agent is available\n        a.threadId = this.threadId() || this.generatedThreadId;\n        if (!this.hasConnectedOnce) {\n          this.hasConnectedOnce = true;\n          if (\"isCopilotKitAgent\" in (a as any)) {\n            this.connectToAgent(a);\n          } else {\n            // Non-CopilotKit agent: nothing to connect; keep default cursor state\n          }\n        }\n      },\n      { allowSignalWrites: true }\n    );\n\n    // Keep agent threadId in sync with input\n    effect(() => {\n      const a = this.agentStore()?.agent;\n      if (a) {\n        a.threadId = this.threadId() || this.generatedThreadId;\n      }\n    });\n  }\n\n  private async connectToAgent(agent: AbstractAgent): Promise<void> {\n    if (!agent) return;\n\n    this.showCursor.set(true);\n    this.cdr.markForCheck();\n\n    try {\n      await agent.runAgent(\n        { forwardedProps: { __copilotkitConnect: true } },\n        {\n          onTextMessageStartEvent: () => {\n            this.showCursor.set(false);\n            this.cdr.detectChanges();\n          },\n          onToolCallStartEvent: () => {\n            this.showCursor.set(false);\n            this.cdr.detectChanges();\n          },\n        }\n      );\n      this.showCursor.set(false);\n      this.cdr.markForCheck();\n    } catch (error) {\n      console.error(\"Failed to connect to agent:\", error);\n      this.showCursor.set(false);\n      this.cdr.markForCheck();\n    }\n  }\n\n  async submitInput(value: string): Promise<void> {\n    const agent = this.agentStore()?.agent;\n    if (!agent || !value.trim()) return;\n\n    // Add user message\n    const userMessage: Message = {\n      id: randomUUID(),\n      role: \"user\",\n      content: value,\n    };\n    agent.addMessage(userMessage);\n\n    // Clear the input\n    this.inputValue.set(\"\");\n\n    // Show cursor while processing\n    this.showCursor.set(true);\n    this.cdr.markForCheck();\n\n    // Run the agent with named subscriber callbacks\n    try {\n      await agent.runAgent(\n        {},\n        {\n          onTextMessageStartEvent: () => {\n            this.showCursor.set(false);\n            this.cdr.detectChanges();\n          },\n          onToolCallStartEvent: () => {\n            this.showCursor.set(false);\n            this.cdr.detectChanges();\n          },\n        }\n      );\n    } catch (error) {\n      console.error(\"Agent run error:\", error);\n    } finally {\n      this.showCursor.set(false);\n      this.cdr.markForCheck();\n    }\n  }\n\n  changeInput(value: string): void {\n    this.inputValue.set(value);\n  }\n}\n"]}
|
|
@@ -0,0 +1,9 @@
|
|
|
1
|
+
import { inject, InjectionToken } from "@angular/core";
|
|
2
|
+
export const COPILOT_KIT_CONFIG = new InjectionToken("COPILOT_KIT_CONFIG");
|
|
3
|
+
export function injectCopilotKitConfig() {
|
|
4
|
+
return inject(COPILOT_KIT_CONFIG);
|
|
5
|
+
}
|
|
6
|
+
export function provideCopilotKit(config) {
|
|
7
|
+
return { provide: COPILOT_KIT_CONFIG, useValue: config };
|
|
8
|
+
}
|
|
9
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29uZmlnLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL2xpYi9jb25maWcudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLE1BQU0sRUFBRSxjQUFjLEVBQVksTUFBTSxlQUFlLENBQUM7QUFvQmpFLE1BQU0sQ0FBQyxNQUFNLGtCQUFrQixHQUFHLElBQUksY0FBYyxDQUNsRCxvQkFBb0IsQ0FDckIsQ0FBQztBQUVGLE1BQU0sVUFBVSxzQkFBc0I7SUFDcEMsT0FBTyxNQUFNLENBQUMsa0JBQWtCLENBQUMsQ0FBQztBQUNwQyxDQUFDO0FBRUQsTUFBTSxVQUFVLGlCQUFpQixDQUFDLE1BQXdCO0lBQ3hELE9BQU8sRUFBRSxPQUFPLEVBQUUsa0JBQWtCLEVBQUUsUUFBUSxFQUFFLE1BQU0sRUFBRSxDQUFDO0FBQzNELENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBpbmplY3QsIEluamVjdGlvblRva2VuLCBQcm92aWRlciB9IGZyb20gXCJAYW5ndWxhci9jb3JlXCI7XG5pbXBvcnQgeyBBYnN0cmFjdEFnZW50IH0gZnJvbSBcIkBhZy11aS9jbGllbnRcIjtcbmltcG9ydCB7XG4gIENsaWVudFRvb2wsXG4gIEZyb250ZW5kVG9vbENvbmZpZyxcbiAgSHVtYW5JblRoZUxvb3BDb25maWcsXG4gIFJlbmRlclRvb2xDYWxsQ29uZmlnLFxufSBmcm9tIFwiLi90b29sc1wiO1xuXG5leHBvcnQgaW50ZXJmYWNlIENvcGlsb3RLaXRDb25maWcge1xuICBydW50aW1lVXJsPzogc3RyaW5nO1xuICBoZWFkZXJzPzogUmVjb3JkPHN0cmluZywgc3RyaW5nPjtcbiAgcHJvcGVydGllcz86IFJlY29yZDxzdHJpbmcsIHVua25vd24+O1xuICBhZ2VudHM/OiBSZWNvcmQ8c3RyaW5nLCBBYnN0cmFjdEFnZW50PjtcbiAgdG9vbHM/OiBDbGllbnRUb29sW107XG4gIHJlbmRlclRvb2xDYWxscz86IFJlbmRlclRvb2xDYWxsQ29uZmlnW107XG4gIGZyb250ZW5kVG9vbHM/OiBGcm9udGVuZFRvb2xDb25maWdbXTtcbiAgaHVtYW5JblRoZUxvb3A/OiBIdW1hbkluVGhlTG9vcENvbmZpZ1tdO1xufVxuXG5leHBvcnQgY29uc3QgQ09QSUxPVF9LSVRfQ09ORklHID0gbmV3IEluamVjdGlvblRva2VuPENvcGlsb3RLaXRDb25maWc+KFxuICBcIkNPUElMT1RfS0lUX0NPTkZJR1wiXG4pO1xuXG5leHBvcnQgZnVuY3Rpb24gaW5qZWN0Q29waWxvdEtpdENvbmZpZygpOiBDb3BpbG90S2l0Q29uZmlnIHtcbiAgcmV0dXJuIGluamVjdChDT1BJTE9UX0tJVF9DT05GSUcpO1xufVxuXG5leHBvcnQgZnVuY3Rpb24gcHJvdmlkZUNvcGlsb3RLaXQoY29uZmlnOiBDb3BpbG90S2l0Q29uZmlnKTogUHJvdmlkZXIge1xuICByZXR1cm4geyBwcm92aWRlOiBDT1BJTE9UX0tJVF9DT05GSUcsIHVzZVZhbHVlOiBjb25maWcgfTtcbn1cbiJdfQ==
|
|
@@ -0,0 +1,115 @@
|
|
|
1
|
+
import { CopilotKitCore } from "@copilotkitnext/core";
|
|
2
|
+
import { Injectable, Injector, runInInjectionContext, signal, inject } from "@angular/core";
|
|
3
|
+
import { injectCopilotKitConfig } from "./config";
|
|
4
|
+
import { HumanInTheLoop } from "./human-in-the-loop";
|
|
5
|
+
import * as i0 from "@angular/core";
|
|
6
|
+
export class CopilotKit {
|
|
7
|
+
#config = injectCopilotKitConfig();
|
|
8
|
+
#hitl = inject(HumanInTheLoop);
|
|
9
|
+
#rootInjector = inject(Injector);
|
|
10
|
+
#agents = signal(this.#config.agents ?? {});
|
|
11
|
+
agents = this.#agents.asReadonly();
|
|
12
|
+
core = new CopilotKitCore({
|
|
13
|
+
runtimeUrl: this.#config.runtimeUrl,
|
|
14
|
+
headers: this.#config.headers,
|
|
15
|
+
properties: this.#config.properties,
|
|
16
|
+
agents__unsafe_dev_only: this.#config.agents,
|
|
17
|
+
tools: this.#config.tools,
|
|
18
|
+
});
|
|
19
|
+
#toolCallRenderConfigs = signal([]);
|
|
20
|
+
#clientToolCallRenderConfigs = signal([]);
|
|
21
|
+
#humanInTheLoopToolRenderConfigs = signal([]);
|
|
22
|
+
toolCallRenderConfigs = this.#toolCallRenderConfigs.asReadonly();
|
|
23
|
+
clientToolCallRenderConfigs = this.#clientToolCallRenderConfigs.asReadonly();
|
|
24
|
+
humanInTheLoopToolRenderConfigs = this.#humanInTheLoopToolRenderConfigs.asReadonly();
|
|
25
|
+
constructor() {
|
|
26
|
+
this.#config.renderToolCalls?.forEach((renderConfig) => {
|
|
27
|
+
this.addRenderToolCall(renderConfig);
|
|
28
|
+
});
|
|
29
|
+
this.#config.tools?.forEach((tool) => {
|
|
30
|
+
if (tool.renderer && tool.parameters) {
|
|
31
|
+
this.addRenderToolCall({
|
|
32
|
+
name: tool.name,
|
|
33
|
+
args: tool.parameters,
|
|
34
|
+
component: tool.renderer,
|
|
35
|
+
agentId: tool.agentId,
|
|
36
|
+
});
|
|
37
|
+
}
|
|
38
|
+
});
|
|
39
|
+
this.#config.frontendTools?.forEach((clientTool) => {
|
|
40
|
+
this.addFrontendTool({ ...clientTool, injector: this.#rootInjector });
|
|
41
|
+
});
|
|
42
|
+
this.#config.humanInTheLoop?.forEach((humanInTheLoopTool) => {
|
|
43
|
+
this.addHumanInTheLoop(humanInTheLoopTool);
|
|
44
|
+
});
|
|
45
|
+
this.core.subscribe({
|
|
46
|
+
onAgentsChanged: () => {
|
|
47
|
+
this.#agents.set(this.core.agents);
|
|
48
|
+
},
|
|
49
|
+
});
|
|
50
|
+
}
|
|
51
|
+
#bindClientTool(clientToolWithInjector) {
|
|
52
|
+
const { injector, handler, ...frontendCandidate } = clientToolWithInjector;
|
|
53
|
+
return {
|
|
54
|
+
...frontendCandidate,
|
|
55
|
+
handler: (args) => runInInjectionContext(injector, () => handler(args)),
|
|
56
|
+
};
|
|
57
|
+
}
|
|
58
|
+
addFrontendTool(clientToolWithInjector) {
|
|
59
|
+
const tool = this.#bindClientTool(clientToolWithInjector);
|
|
60
|
+
this.core.addTool(tool);
|
|
61
|
+
this.#clientToolCallRenderConfigs.update((current) => [...current, clientToolWithInjector]);
|
|
62
|
+
}
|
|
63
|
+
addRenderToolCall(renderConfig) {
|
|
64
|
+
this.#toolCallRenderConfigs.update((current) => [...current, renderConfig]);
|
|
65
|
+
}
|
|
66
|
+
#bindHumanInTheLoopTool(humanInTheLoopTool) {
|
|
67
|
+
return {
|
|
68
|
+
...humanInTheLoopTool,
|
|
69
|
+
handler: (args, toolCall) => {
|
|
70
|
+
return this.#hitl.onResult(toolCall.id, humanInTheLoopTool.name);
|
|
71
|
+
},
|
|
72
|
+
};
|
|
73
|
+
}
|
|
74
|
+
addHumanInTheLoop(humanInTheLoopTool) {
|
|
75
|
+
this.#humanInTheLoopToolRenderConfigs.update((current) => [...current, humanInTheLoopTool]);
|
|
76
|
+
const tool = this.#bindHumanInTheLoopTool(humanInTheLoopTool);
|
|
77
|
+
this.core.addTool(tool);
|
|
78
|
+
}
|
|
79
|
+
#isSameAgentId(target, agentId) {
|
|
80
|
+
if (agentId) {
|
|
81
|
+
return target.agentId === agentId;
|
|
82
|
+
}
|
|
83
|
+
return true;
|
|
84
|
+
}
|
|
85
|
+
removeTool(toolName, agentId) {
|
|
86
|
+
this.core.removeTool(toolName);
|
|
87
|
+
this.#clientToolCallRenderConfigs.update((current) => current.filter((renderConfig) => renderConfig.name !== toolName && this.#isSameAgentId(renderConfig, agentId)));
|
|
88
|
+
this.#humanInTheLoopToolRenderConfigs.update((current) => current.filter((renderConfig) => renderConfig.name !== toolName && this.#isSameAgentId(renderConfig, agentId)));
|
|
89
|
+
this.#toolCallRenderConfigs.update((current) => current.filter((renderConfig) => renderConfig.name !== toolName && this.#isSameAgentId(renderConfig, agentId)));
|
|
90
|
+
}
|
|
91
|
+
getAgent(agentId) {
|
|
92
|
+
return this.core.getAgent(agentId);
|
|
93
|
+
}
|
|
94
|
+
updateRuntime(options) {
|
|
95
|
+
if (options.runtimeUrl !== undefined) {
|
|
96
|
+
this.core.setRuntimeUrl(options.runtimeUrl);
|
|
97
|
+
}
|
|
98
|
+
if (options.headers !== undefined) {
|
|
99
|
+
this.core.setHeaders(options.headers);
|
|
100
|
+
}
|
|
101
|
+
if (options.properties !== undefined) {
|
|
102
|
+
this.core.setProperties(options.properties);
|
|
103
|
+
}
|
|
104
|
+
if (options.agents !== undefined) {
|
|
105
|
+
this.core.setAgents__unsafe_dev_only(options.agents);
|
|
106
|
+
}
|
|
107
|
+
}
|
|
108
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: CopilotKit, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
|
|
109
|
+
static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: CopilotKit, providedIn: "root" });
|
|
110
|
+
}
|
|
111
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: CopilotKit, decorators: [{
|
|
112
|
+
type: Injectable,
|
|
113
|
+
args: [{ providedIn: "root" }]
|
|
114
|
+
}], ctorParameters: () => [] });
|
|
115
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"copilotkit.js","sourceRoot":"","sources":["../../../src/lib/copilotkit.ts"],"names":[],"mappings":"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","sourcesContent":["import { AbstractAgent } from \"@ag-ui/client\";\nimport { FrontendTool, CopilotKitCore } from \"@copilotkitnext/core\";\nimport { Injectable, Injector, Signal, WritableSignal, runInInjectionContext, signal, inject } from \"@angular/core\";\nimport { FrontendToolConfig, HumanInTheLoopConfig, RenderToolCallConfig } from \"./tools\";\nimport { injectCopilotKitConfig } from \"./config\";\nimport { HumanInTheLoop } from \"./human-in-the-loop\";\n\n@Injectable({ providedIn: \"root\" })\nexport class CopilotKit {\n  readonly #config = injectCopilotKitConfig();\n  readonly #hitl = inject(HumanInTheLoop);\n  readonly #rootInjector = inject(Injector);\n  readonly #agents = signal<Record<string, AbstractAgent>>(this.#config.agents ?? {});\n  readonly agents = this.#agents.asReadonly();\n\n  readonly core = new CopilotKitCore({\n    runtimeUrl: this.#config.runtimeUrl,\n    headers: this.#config.headers,\n    properties: this.#config.properties,\n    agents__unsafe_dev_only: this.#config.agents,\n    tools: this.#config.tools,\n  });\n\n  readonly #toolCallRenderConfigs: WritableSignal<RenderToolCallConfig[]> = signal([]);\n  readonly #clientToolCallRenderConfigs: WritableSignal<FrontendToolConfig[]> = signal([]);\n  readonly #humanInTheLoopToolRenderConfigs: WritableSignal<HumanInTheLoopConfig[]> = signal([]);\n\n  readonly toolCallRenderConfigs: Signal<RenderToolCallConfig[]> = this.#toolCallRenderConfigs.asReadonly();\n  readonly clientToolCallRenderConfigs: Signal<FrontendToolConfig[]> = this.#clientToolCallRenderConfigs.asReadonly();\n  readonly humanInTheLoopToolRenderConfigs: Signal<HumanInTheLoopConfig[]> =\n    this.#humanInTheLoopToolRenderConfigs.asReadonly();\n\n  constructor() {\n    this.#config.renderToolCalls?.forEach((renderConfig) => {\n      this.addRenderToolCall(renderConfig);\n    });\n\n    this.#config.tools?.forEach((tool) => {\n      if (tool.renderer && tool.parameters) {\n        this.addRenderToolCall({\n          name: tool.name,\n          args: tool.parameters,\n          component: tool.renderer,\n          agentId: tool.agentId,\n        });\n      }\n    });\n\n    this.#config.frontendTools?.forEach((clientTool) => {\n      this.addFrontendTool({ ...clientTool, injector: this.#rootInjector });\n    });\n\n    this.#config.humanInTheLoop?.forEach((humanInTheLoopTool) => {\n      this.addHumanInTheLoop(humanInTheLoopTool);\n    });\n\n    this.core.subscribe({\n      onAgentsChanged: () => {\n        this.#agents.set(this.core.agents);\n      },\n    });\n  }\n\n  #bindClientTool(\n    clientToolWithInjector: FrontendToolConfig & {\n      injector: Injector;\n    },\n  ): FrontendTool {\n    const { injector, handler, ...frontendCandidate } = clientToolWithInjector;\n\n    return {\n      ...frontendCandidate,\n      handler: (args) => runInInjectionContext(injector, () => handler(args)),\n    };\n  }\n\n  addFrontendTool(\n    clientToolWithInjector: FrontendToolConfig & {\n      injector: Injector;\n    },\n  ): void {\n    const tool = this.#bindClientTool(clientToolWithInjector);\n\n    this.core.addTool(tool);\n\n    this.#clientToolCallRenderConfigs.update((current) => [...current, clientToolWithInjector]);\n  }\n\n  addRenderToolCall(renderConfig: RenderToolCallConfig): void {\n    this.#toolCallRenderConfigs.update((current) => [...current, renderConfig]);\n  }\n\n  #bindHumanInTheLoopTool(humanInTheLoopTool: HumanInTheLoopConfig): FrontendTool {\n    return {\n      ...humanInTheLoopTool,\n      handler: (args, toolCall) => {\n        return this.#hitl.onResult(toolCall.id, humanInTheLoopTool.name);\n      },\n    };\n  }\n\n  addHumanInTheLoop(humanInTheLoopTool: HumanInTheLoopConfig): void {\n    this.#humanInTheLoopToolRenderConfigs.update((current) => [...current, humanInTheLoopTool]);\n\n    const tool = this.#bindHumanInTheLoopTool(humanInTheLoopTool);\n\n    this.core.addTool(tool);\n  }\n\n  #isSameAgentId<T extends { agentId?: string }>(target: T, agentId?: string): boolean {\n    if (agentId) {\n      return target.agentId === agentId;\n    }\n\n    return true;\n  }\n\n  removeTool(toolName: string, agentId?: string): void {\n    this.core.removeTool(toolName);\n    this.#clientToolCallRenderConfigs.update((current) =>\n      current.filter((renderConfig) => renderConfig.name !== toolName && this.#isSameAgentId(renderConfig, agentId)),\n    );\n    this.#humanInTheLoopToolRenderConfigs.update((current) =>\n      current.filter((renderConfig) => renderConfig.name !== toolName && this.#isSameAgentId(renderConfig, agentId)),\n    );\n    this.#toolCallRenderConfigs.update((current) =>\n      current.filter((renderConfig) => renderConfig.name !== toolName && this.#isSameAgentId(renderConfig, agentId)),\n    );\n  }\n\n  getAgent(agentId: string): AbstractAgent | undefined {\n    return this.core.getAgent(agentId);\n  }\n\n  updateRuntime(options: {\n    runtimeUrl?: string;\n    headers?: Record<string, string>;\n    properties?: Record<string, unknown>;\n    agents?: Record<string, AbstractAgent>;\n  }): void {\n    if (options.runtimeUrl !== undefined) {\n      this.core.setRuntimeUrl(options.runtimeUrl);\n    }\n    if (options.headers !== undefined) {\n      this.core.setHeaders(options.headers);\n    }\n    if (options.properties !== undefined) {\n      this.core.setProperties(options.properties);\n    }\n    if (options.agents !== undefined) {\n      this.core.setAgents__unsafe_dev_only(options.agents);\n    }\n  }\n}\n"]}
|
|
@@ -0,0 +1,130 @@
|
|
|
1
|
+
import { Directive, Input, Inject, } from "@angular/core";
|
|
2
|
+
import { CopilotKit } from "../copilotkit";
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
import * as i1 from "../copilotkit";
|
|
5
|
+
/**
|
|
6
|
+
* Directive to manage agent context in CopilotKit.
|
|
7
|
+
* Automatically adds context on init, updates on changes, and removes on destroy.
|
|
8
|
+
*
|
|
9
|
+
* @example
|
|
10
|
+
* ```html
|
|
11
|
+
* <!-- With separate inputs -->
|
|
12
|
+
* <div copilotkitAgentContext
|
|
13
|
+
* [description]="'User preferences'"
|
|
14
|
+
* [value]="userSettings">
|
|
15
|
+
* </div>
|
|
16
|
+
*
|
|
17
|
+
* <!-- With context object -->
|
|
18
|
+
* <div [copilotkitAgentContext]="contextObject">
|
|
19
|
+
* </div>
|
|
20
|
+
*
|
|
21
|
+
* <!-- With dynamic values -->
|
|
22
|
+
* <div copilotkitAgentContext
|
|
23
|
+
* description="Form state"
|
|
24
|
+
* [value]="formData$ | async">
|
|
25
|
+
* </div>
|
|
26
|
+
* ```
|
|
27
|
+
*/
|
|
28
|
+
export class CopilotKitAgentContext {
|
|
29
|
+
copilotkit;
|
|
30
|
+
contextId;
|
|
31
|
+
constructor(copilotkit) {
|
|
32
|
+
this.copilotkit = copilotkit;
|
|
33
|
+
}
|
|
34
|
+
/**
|
|
35
|
+
* Context object containing both description and value.
|
|
36
|
+
* If provided, this takes precedence over individual inputs.
|
|
37
|
+
*/
|
|
38
|
+
context;
|
|
39
|
+
/**
|
|
40
|
+
* Description of the context.
|
|
41
|
+
* Used when context object is not provided.
|
|
42
|
+
*/
|
|
43
|
+
description;
|
|
44
|
+
/**
|
|
45
|
+
* Value of the context.
|
|
46
|
+
* Used when context object is not provided.
|
|
47
|
+
*/
|
|
48
|
+
value;
|
|
49
|
+
ngOnInit() {
|
|
50
|
+
this.addContext();
|
|
51
|
+
}
|
|
52
|
+
ngOnChanges(changes) {
|
|
53
|
+
// Check if any relevant input has changed
|
|
54
|
+
const hasContextChange = "context" in changes;
|
|
55
|
+
const hasDescriptionChange = "description" in changes;
|
|
56
|
+
const hasValueChange = "value" in changes;
|
|
57
|
+
if (hasContextChange || hasDescriptionChange || hasValueChange) {
|
|
58
|
+
// Skip the first change as ngOnInit handles initial setup
|
|
59
|
+
if (this.contextId) {
|
|
60
|
+
this.updateContext();
|
|
61
|
+
}
|
|
62
|
+
}
|
|
63
|
+
}
|
|
64
|
+
ngOnDestroy() {
|
|
65
|
+
this.removeContext();
|
|
66
|
+
}
|
|
67
|
+
/**
|
|
68
|
+
* Adds the context to CopilotKit
|
|
69
|
+
*/
|
|
70
|
+
addContext() {
|
|
71
|
+
const contextToAdd = this.getContext();
|
|
72
|
+
if (contextToAdd) {
|
|
73
|
+
this.contextId = this.copilotkit.core.addContext(contextToAdd);
|
|
74
|
+
}
|
|
75
|
+
}
|
|
76
|
+
/**
|
|
77
|
+
* Updates the context by removing the old one and adding a new one
|
|
78
|
+
*/
|
|
79
|
+
updateContext() {
|
|
80
|
+
this.removeContext();
|
|
81
|
+
this.addContext();
|
|
82
|
+
}
|
|
83
|
+
/**
|
|
84
|
+
* Removes the current context from CopilotKit
|
|
85
|
+
*/
|
|
86
|
+
removeContext() {
|
|
87
|
+
if (this.contextId) {
|
|
88
|
+
this.copilotkit.core.removeContext(this.contextId);
|
|
89
|
+
this.contextId = undefined;
|
|
90
|
+
}
|
|
91
|
+
}
|
|
92
|
+
/**
|
|
93
|
+
* Gets the context object from inputs
|
|
94
|
+
*/
|
|
95
|
+
getContext() {
|
|
96
|
+
// If context object is provided, use it
|
|
97
|
+
if (this.context) {
|
|
98
|
+
return this.context;
|
|
99
|
+
}
|
|
100
|
+
// Otherwise, build from individual inputs
|
|
101
|
+
// Note: null is a valid value, but undefined means not set
|
|
102
|
+
if (this.description !== undefined && this.value !== undefined) {
|
|
103
|
+
return {
|
|
104
|
+
description: this.description,
|
|
105
|
+
value: this.value,
|
|
106
|
+
};
|
|
107
|
+
}
|
|
108
|
+
return null;
|
|
109
|
+
}
|
|
110
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: CopilotKitAgentContext, deps: [{ token: CopilotKit }], target: i0.ɵɵFactoryTarget.Directive });
|
|
111
|
+
static ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "18.2.14", type: CopilotKitAgentContext, isStandalone: true, selector: "[copilotkitAgentContext]", inputs: { context: ["copilotkitAgentContext", "context"], description: "description", value: "value" }, usesOnChanges: true, ngImport: i0 });
|
|
112
|
+
}
|
|
113
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: CopilotKitAgentContext, decorators: [{
|
|
114
|
+
type: Directive,
|
|
115
|
+
args: [{
|
|
116
|
+
selector: "[copilotkitAgentContext]",
|
|
117
|
+
standalone: true,
|
|
118
|
+
}]
|
|
119
|
+
}], ctorParameters: () => [{ type: i1.CopilotKit, decorators: [{
|
|
120
|
+
type: Inject,
|
|
121
|
+
args: [CopilotKit]
|
|
122
|
+
}] }], propDecorators: { context: [{
|
|
123
|
+
type: Input,
|
|
124
|
+
args: ["copilotkitAgentContext"]
|
|
125
|
+
}], description: [{
|
|
126
|
+
type: Input
|
|
127
|
+
}], value: [{
|
|
128
|
+
type: Input
|
|
129
|
+
}] } });
|
|
130
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29waWxvdGtpdC1hZ2VudC1jb250ZXh0LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vc3JjL2xpYi9kaXJlY3RpdmVzL2NvcGlsb3RraXQtYWdlbnQtY29udGV4dC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQ0wsU0FBUyxFQUNULEtBQUssRUFLTCxNQUFNLEdBQ1AsTUFBTSxlQUFlLENBQUM7QUFDdkIsT0FBTyxFQUFFLFVBQVUsRUFBRSxNQUFNLGVBQWUsQ0FBQzs7O0FBRzNDOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0dBc0JHO0FBS0gsTUFBTSxPQUFPLHNCQUFzQjtJQUdnQjtJQUZ6QyxTQUFTLENBQVU7SUFFM0IsWUFBaUQsVUFBc0I7UUFBdEIsZUFBVSxHQUFWLFVBQVUsQ0FBWTtJQUFHLENBQUM7SUFFM0U7OztPQUdHO0lBQzhCLE9BQU8sQ0FBVztJQUVuRDs7O09BR0c7SUFDTSxXQUFXLENBQVU7SUFFOUI7OztPQUdHO0lBQ00sS0FBSyxDQUFPO0lBRXJCLFFBQVE7UUFDTixJQUFJLENBQUMsVUFBVSxFQUFFLENBQUM7SUFDcEIsQ0FBQztJQUVELFdBQVcsQ0FBQyxPQUFzQjtRQUNoQywwQ0FBMEM7UUFDMUMsTUFBTSxnQkFBZ0IsR0FBRyxTQUFTLElBQUksT0FBTyxDQUFDO1FBQzlDLE1BQU0sb0JBQW9CLEdBQUcsYUFBYSxJQUFJLE9BQU8sQ0FBQztRQUN0RCxNQUFNLGNBQWMsR0FBRyxPQUFPLElBQUksT0FBTyxDQUFDO1FBRTFDLElBQUksZ0JBQWdCLElBQUksb0JBQW9CLElBQUksY0FBYyxFQUFFLENBQUM7WUFDL0QsMERBQTBEO1lBQzFELElBQUksSUFBSSxDQUFDLFNBQVMsRUFBRSxDQUFDO2dCQUNuQixJQUFJLENBQUMsYUFBYSxFQUFFLENBQUM7WUFDdkIsQ0FBQztRQUNILENBQUM7SUFDSCxDQUFDO0lBRUQsV0FBVztRQUNULElBQUksQ0FBQyxhQUFhLEVBQUUsQ0FBQztJQUN2QixDQUFDO0lBRUQ7O09BRUc7SUFDSyxVQUFVO1FBQ2hCLE1BQU0sWUFBWSxHQUFHLElBQUksQ0FBQyxVQUFVLEVBQUUsQ0FBQztRQUV2QyxJQUFJLFlBQVksRUFBRSxDQUFDO1lBQ2pCLElBQUksQ0FBQyxTQUFTLEdBQUcsSUFBSSxDQUFDLFVBQVUsQ0FBQyxJQUFJLENBQUMsVUFBVSxDQUFDLFlBQVksQ0FBQyxDQUFDO1FBQ2pFLENBQUM7SUFDSCxDQUFDO0lBRUQ7O09BRUc7SUFDSyxhQUFhO1FBQ25CLElBQUksQ0FBQyxhQUFhLEVBQUUsQ0FBQztRQUNyQixJQUFJLENBQUMsVUFBVSxFQUFFLENBQUM7SUFDcEIsQ0FBQztJQUVEOztPQUVHO0lBQ0ssYUFBYTtRQUNuQixJQUFJLElBQUksQ0FBQyxTQUFTLEVBQUUsQ0FBQztZQUNuQixJQUFJLENBQUMsVUFBVSxDQUFDLElBQUksQ0FBQyxhQUFhLENBQUMsSUFBSSxDQUFDLFNBQVMsQ0FBQyxDQUFDO1lBQ25ELElBQUksQ0FBQyxTQUFTLEdBQUcsU0FBUyxDQUFDO1FBQzdCLENBQUM7SUFDSCxDQUFDO0lBRUQ7O09BRUc7SUFDSyxVQUFVO1FBQ2hCLHdDQUF3QztRQUN4QyxJQUFJLElBQUksQ0FBQyxPQUFPLEVBQUUsQ0FBQztZQUNqQixPQUFPLElBQUksQ0FBQyxPQUFPLENBQUM7UUFDdEIsQ0FBQztRQUVELDBDQUEwQztRQUMxQywyREFBMkQ7UUFDM0QsSUFBSSxJQUFJLENBQUMsV0FBVyxLQUFLLFNBQVMsSUFBSSxJQUFJLENBQUMsS0FBSyxLQUFLLFNBQVMsRUFBRSxDQUFDO1lBQy9ELE9BQU87Z0JBQ0wsV0FBVyxFQUFFLElBQUksQ0FBQyxXQUFXO2dCQUM3QixLQUFLLEVBQUUsSUFBSSxDQUFDLEtBQUs7YUFDbEIsQ0FBQztRQUNKLENBQUM7UUFFRCxPQUFPLElBQUksQ0FBQztJQUNkLENBQUM7d0dBN0ZVLHNCQUFzQixrQkFHYixVQUFVOzRGQUhuQixzQkFBc0I7OzRGQUF0QixzQkFBc0I7a0JBSmxDLFNBQVM7bUJBQUM7b0JBQ1QsUUFBUSxFQUFFLDBCQUEwQjtvQkFDcEMsVUFBVSxFQUFFLElBQUk7aUJBQ2pCOzswQkFJYyxNQUFNOzJCQUFDLFVBQVU7eUNBTUcsT0FBTztzQkFBdkMsS0FBSzt1QkFBQyx3QkFBd0I7Z0JBTXRCLFdBQVc7c0JBQW5CLEtBQUs7Z0JBTUcsS0FBSztzQkFBYixLQUFLIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHtcbiAgRGlyZWN0aXZlLFxuICBJbnB1dCxcbiAgT25Jbml0LFxuICBPbkNoYW5nZXMsXG4gIE9uRGVzdHJveSxcbiAgU2ltcGxlQ2hhbmdlcyxcbiAgSW5qZWN0LFxufSBmcm9tIFwiQGFuZ3VsYXIvY29yZVwiO1xuaW1wb3J0IHsgQ29waWxvdEtpdCB9IGZyb20gXCIuLi9jb3BpbG90a2l0XCI7XG5pbXBvcnQgdHlwZSB7IENvbnRleHQgfSBmcm9tIFwiQGFnLXVpL2NsaWVudFwiO1xuXG4vKipcbiAqIERpcmVjdGl2ZSB0byBtYW5hZ2UgYWdlbnQgY29udGV4dCBpbiBDb3BpbG90S2l0LlxuICogQXV0b21hdGljYWxseSBhZGRzIGNvbnRleHQgb24gaW5pdCwgdXBkYXRlcyBvbiBjaGFuZ2VzLCBhbmQgcmVtb3ZlcyBvbiBkZXN0cm95LlxuICpcbiAqIEBleGFtcGxlXG4gKiBgYGBodG1sXG4gKiA8IS0tIFdpdGggc2VwYXJhdGUgaW5wdXRzIC0tPlxuICogPGRpdiBjb3BpbG90a2l0QWdlbnRDb250ZXh0XG4gKiAgICAgIFtkZXNjcmlwdGlvbl09XCInVXNlciBwcmVmZXJlbmNlcydcIlxuICogICAgICBbdmFsdWVdPVwidXNlclNldHRpbmdzXCI+XG4gKiA8L2Rpdj5cbiAqXG4gKiA8IS0tIFdpdGggY29udGV4dCBvYmplY3QgLS0+XG4gKiA8ZGl2IFtjb3BpbG90a2l0QWdlbnRDb250ZXh0XT1cImNvbnRleHRPYmplY3RcIj5cbiAqIDwvZGl2PlxuICpcbiAqIDwhLS0gV2l0aCBkeW5hbWljIHZhbHVlcyAtLT5cbiAqIDxkaXYgY29waWxvdGtpdEFnZW50Q29udGV4dFxuICogICAgICBkZXNjcmlwdGlvbj1cIkZvcm0gc3RhdGVcIlxuICogICAgICBbdmFsdWVdPVwiZm9ybURhdGEkIHwgYXN5bmNcIj5cbiAqIDwvZGl2PlxuICogYGBgXG4gKi9cbkBEaXJlY3RpdmUoe1xuICBzZWxlY3RvcjogXCJbY29waWxvdGtpdEFnZW50Q29udGV4dF1cIixcbiAgc3RhbmRhbG9uZTogdHJ1ZSxcbn0pXG5leHBvcnQgY2xhc3MgQ29waWxvdEtpdEFnZW50Q29udGV4dCBpbXBsZW1lbnRzIE9uSW5pdCwgT25DaGFuZ2VzLCBPbkRlc3Ryb3kge1xuICBwcml2YXRlIGNvbnRleHRJZD86IHN0cmluZztcblxuICBjb25zdHJ1Y3RvcihASW5qZWN0KENvcGlsb3RLaXQpIHByaXZhdGUgcmVhZG9ubHkgY29waWxvdGtpdDogQ29waWxvdEtpdCkge31cblxuICAvKipcbiAgICogQ29udGV4dCBvYmplY3QgY29udGFpbmluZyBib3RoIGRlc2NyaXB0aW9uIGFuZCB2YWx1ZS5cbiAgICogSWYgcHJvdmlkZWQsIHRoaXMgdGFrZXMgcHJlY2VkZW5jZSBvdmVyIGluZGl2aWR1YWwgaW5wdXRzLlxuICAgKi9cbiAgQElucHV0KFwiY29waWxvdGtpdEFnZW50Q29udGV4dFwiKSBjb250ZXh0PzogQ29udGV4dDtcblxuICAvKipcbiAgICogRGVzY3JpcHRpb24gb2YgdGhlIGNvbnRleHQuXG4gICAqIFVzZWQgd2hlbiBjb250ZXh0IG9iamVjdCBpcyBub3QgcHJvdmlkZWQuXG4gICAqL1xuICBASW5wdXQoKSBkZXNjcmlwdGlvbj86IHN0cmluZztcblxuICAvKipcbiAgICogVmFsdWUgb2YgdGhlIGNvbnRleHQuXG4gICAqIFVzZWQgd2hlbiBjb250ZXh0IG9iamVjdCBpcyBub3QgcHJvdmlkZWQuXG4gICAqL1xuICBASW5wdXQoKSB2YWx1ZT86IGFueTtcblxuICBuZ09uSW5pdCgpOiB2b2lkIHtcbiAgICB0aGlzLmFkZENvbnRleHQoKTtcbiAgfVxuXG4gIG5nT25DaGFuZ2VzKGNoYW5nZXM6IFNpbXBsZUNoYW5nZXMpOiB2b2lkIHtcbiAgICAvLyBDaGVjayBpZiBhbnkgcmVsZXZhbnQgaW5wdXQgaGFzIGNoYW5nZWRcbiAgICBjb25zdCBoYXNDb250ZXh0Q2hhbmdlID0gXCJjb250ZXh0XCIgaW4gY2hhbmdlcztcbiAgICBjb25zdCBoYXNEZXNjcmlwdGlvbkNoYW5nZSA9IFwiZGVzY3JpcHRpb25cIiBpbiBjaGFuZ2VzO1xuICAgIGNvbnN0IGhhc1ZhbHVlQ2hhbmdlID0gXCJ2YWx1ZVwiIGluIGNoYW5nZXM7XG5cbiAgICBpZiAoaGFzQ29udGV4dENoYW5nZSB8fCBoYXNEZXNjcmlwdGlvbkNoYW5nZSB8fCBoYXNWYWx1ZUNoYW5nZSkge1xuICAgICAgLy8gU2tpcCB0aGUgZmlyc3QgY2hhbmdlIGFzIG5nT25Jbml0IGhhbmRsZXMgaW5pdGlhbCBzZXR1cFxuICAgICAgaWYgKHRoaXMuY29udGV4dElkKSB7XG4gICAgICAgIHRoaXMudXBkYXRlQ29udGV4dCgpO1xuICAgICAgfVxuICAgIH1cbiAgfVxuXG4gIG5nT25EZXN0cm95KCk6IHZvaWQge1xuICAgIHRoaXMucmVtb3ZlQ29udGV4dCgpO1xuICB9XG5cbiAgLyoqXG4gICAqIEFkZHMgdGhlIGNvbnRleHQgdG8gQ29waWxvdEtpdFxuICAgKi9cbiAgcHJpdmF0ZSBhZGRDb250ZXh0KCk6IHZvaWQge1xuICAgIGNvbnN0IGNvbnRleHRUb0FkZCA9IHRoaXMuZ2V0Q29udGV4dCgpO1xuXG4gICAgaWYgKGNvbnRleHRUb0FkZCkge1xuICAgICAgdGhpcy5jb250ZXh0SWQgPSB0aGlzLmNvcGlsb3RraXQuY29yZS5hZGRDb250ZXh0KGNvbnRleHRUb0FkZCk7XG4gICAgfVxuICB9XG5cbiAgLyoqXG4gICAqIFVwZGF0ZXMgdGhlIGNvbnRleHQgYnkgcmVtb3ZpbmcgdGhlIG9sZCBvbmUgYW5kIGFkZGluZyBhIG5ldyBvbmVcbiAgICovXG4gIHByaXZhdGUgdXBkYXRlQ29udGV4dCgpOiB2b2lkIHtcbiAgICB0aGlzLnJlbW92ZUNvbnRleHQoKTtcbiAgICB0aGlzLmFkZENvbnRleHQoKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBSZW1vdmVzIHRoZSBjdXJyZW50IGNvbnRleHQgZnJvbSBDb3BpbG90S2l0XG4gICAqL1xuICBwcml2YXRlIHJlbW92ZUNvbnRleHQoKTogdm9pZCB7XG4gICAgaWYgKHRoaXMuY29udGV4dElkKSB7XG4gICAgICB0aGlzLmNvcGlsb3RraXQuY29yZS5yZW1vdmVDb250ZXh0KHRoaXMuY29udGV4dElkKTtcbiAgICAgIHRoaXMuY29udGV4dElkID0gdW5kZWZpbmVkO1xuICAgIH1cbiAgfVxuXG4gIC8qKlxuICAgKiBHZXRzIHRoZSBjb250ZXh0IG9iamVjdCBmcm9tIGlucHV0c1xuICAgKi9cbiAgcHJpdmF0ZSBnZXRDb250ZXh0KCk6IENvbnRleHQgfCBudWxsIHtcbiAgICAvLyBJZiBjb250ZXh0IG9iamVjdCBpcyBwcm92aWRlZCwgdXNlIGl0XG4gICAgaWYgKHRoaXMuY29udGV4dCkge1xuICAgICAgcmV0dXJuIHRoaXMuY29udGV4dDtcbiAgICB9XG5cbiAgICAvLyBPdGhlcndpc2UsIGJ1aWxkIGZyb20gaW5kaXZpZHVhbCBpbnB1dHNcbiAgICAvLyBOb3RlOiBudWxsIGlzIGEgdmFsaWQgdmFsdWUsIGJ1dCB1bmRlZmluZWQgbWVhbnMgbm90IHNldFxuICAgIGlmICh0aGlzLmRlc2NyaXB0aW9uICE9PSB1bmRlZmluZWQgJiYgdGhpcy52YWx1ZSAhPT0gdW5kZWZpbmVkKSB7XG4gICAgICByZXR1cm4ge1xuICAgICAgICBkZXNjcmlwdGlvbjogdGhpcy5kZXNjcmlwdGlvbixcbiAgICAgICAgdmFsdWU6IHRoaXMudmFsdWUsXG4gICAgICB9O1xuICAgIH1cblxuICAgIHJldHVybiBudWxsO1xuICB9XG59XG4iXX0=
|