@copilotkitnext/angular 0.0.4 → 0.0.6
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/dist/components/chat/copilot-chat-assistant-message.component.d.ts +1 -0
- package/dist/core/copilotkit.providers.d.ts +1 -1
- package/dist/core/copilotkit.service.d.ts +5 -5
- package/dist/core/copilotkit.types.d.ts +8 -10
- package/dist/directives/copilotkit-frontend-tool.directive.d.ts +1 -1
- package/dist/esm2022/components/chat/copilot-chat-assistant-message.component.mjs +11 -5
- package/dist/esm2022/components/chat/copilot-chat-input.component.mjs +2 -2
- package/dist/esm2022/components/chat/copilot-chat-textarea.component.mjs +2 -2
- package/dist/esm2022/components/chat/copilot-chat-tool-calls-view.component.mjs +14 -8
- package/dist/esm2022/components/chat/copilot-chat.component.mjs +10 -3
- package/dist/esm2022/components/copilotkit-tool-render.component.mjs +7 -10
- package/dist/esm2022/core/copilotkit.providers.mjs +1 -1
- package/dist/esm2022/core/copilotkit.service.mjs +6 -22
- package/dist/esm2022/core/copilotkit.types.mjs +2 -1
- package/dist/esm2022/directives/copilotkit-frontend-tool.directive.mjs +1 -3
- package/dist/esm2022/directives/copilotkit-human-in-the-loop.directive.mjs +1 -2
- package/dist/esm2022/lib/slots/copilot-slot.component.mjs +15 -5
- package/dist/esm2022/lib/slots/slot.utils.mjs +14 -5
- package/dist/esm2022/utils/frontend-tool.utils.mjs +1 -5
- package/dist/esm2022/utils/human-in-the-loop.utils.mjs +4 -7
- package/dist/fesm2022/copilotkitnext-angular.mjs +76 -65
- package/dist/fesm2022/copilotkitnext-angular.mjs.map +1 -1
- package/dist/utils/frontend-tool.utils.d.ts +1 -1
- package/package.json +3 -3
|
@@ -54,10 +54,9 @@ export function registerHumanInTheLoop(tool) {
|
|
|
54
54
|
service.copilotkit.addTool(frontendTool);
|
|
55
55
|
const toolId = frontendTool.name;
|
|
56
56
|
// Register tool render if provided
|
|
57
|
-
if (frontendTool.render
|
|
57
|
+
if (frontendTool.render) {
|
|
58
58
|
service.registerToolRender(frontendTool.name, {
|
|
59
59
|
name: frontendTool.name,
|
|
60
|
-
args: tool.parameters,
|
|
61
60
|
render: frontendTool.render
|
|
62
61
|
});
|
|
63
62
|
}
|
|
@@ -131,10 +130,9 @@ export function addHumanInTheLoop(service, tool) {
|
|
|
131
130
|
service.copilotkit.addTool(frontendTool);
|
|
132
131
|
const toolId = frontendTool.name;
|
|
133
132
|
// Register tool render if provided
|
|
134
|
-
if (frontendTool.render
|
|
133
|
+
if (frontendTool.render) {
|
|
135
134
|
service.registerToolRender(frontendTool.name, {
|
|
136
135
|
name: frontendTool.name,
|
|
137
|
-
args: tool.parameters,
|
|
138
136
|
render: frontendTool.render
|
|
139
137
|
});
|
|
140
138
|
}
|
|
@@ -208,10 +206,9 @@ export function createHumanInTheLoop(service, tool) {
|
|
|
208
206
|
service.copilotkit.addTool(frontendTool);
|
|
209
207
|
toolId = frontendTool.name;
|
|
210
208
|
// Register tool render if provided
|
|
211
|
-
if (frontendTool.render
|
|
209
|
+
if (frontendTool.render) {
|
|
212
210
|
service.registerToolRender(frontendTool.name, {
|
|
213
211
|
name: frontendTool.name,
|
|
214
|
-
args: currentTool.parameters,
|
|
215
212
|
render: frontendTool.render
|
|
216
213
|
});
|
|
217
214
|
}
|
|
@@ -293,4 +290,4 @@ export function enhancePropsForHumanInTheLoop(props, status, respond) {
|
|
|
293
290
|
respond: undefined
|
|
294
291
|
};
|
|
295
292
|
}
|
|
296
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"human-in-the-loop.utils.js","sourceRoot":"","sources":["../../../src/utils/human-in-the-loop.utils.ts"],"names":[],"mappings":"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","sourcesContent":["import { \n  DestroyRef, \n  inject, \n  signal, \n  Signal,\n  Type,\n  TemplateRef\n} from '@angular/core';\nimport { CopilotKitService } from '../core/copilotkit.service';\nimport { \n  AngularHumanInTheLoop, \n  ToolCallStatus,\n  HumanInTheLoopState,\n  HumanInTheLoopProps,\n  AngularFrontendTool\n} from '../core/copilotkit.types';\n\n/**\n * Registers a human-in-the-loop tool that requires user interaction.\n * Must be called within an injection context.\n * Automatically cleans up when the component/service is destroyed.\n * \n * @param tool - The human-in-the-loop tool configuration\n * @returns The tool ID\n * \n * @example\n * ```typescript\n * export class ApprovalComponent {\n *   toolId = registerHumanInTheLoop({\n *     name: 'requireApproval',\n *     description: 'Requires user approval',\n *     args: z.object({ action: z.string() }),\n *     render: ApprovalDialogComponent\n *   });\n * }\n * ```\n */\nexport function registerHumanInTheLoop<T extends Record<string, any> = Record<string, any>>(\n  tool: AngularHumanInTheLoop<T>\n): string {\n  const service = inject(CopilotKitService);\n  const destroyRef = inject(DestroyRef);\n  \n  // Create state management\n  const statusSignal = signal<ToolCallStatus>(ToolCallStatus.InProgress);\n  let resolvePromise: ((result: unknown) => void) | null = null;\n  \n  // Create respond function\n  const respond = async (result: unknown): Promise<void> => {\n    if (resolvePromise) {\n      resolvePromise(result);\n      statusSignal.set(ToolCallStatus.Complete);\n      resolvePromise = null;\n    }\n  };\n  \n  // Create handler that returns a Promise\n  const handler = async (_args: T): Promise<unknown> => {\n    return new Promise((resolve) => {\n      statusSignal.set(ToolCallStatus.Executing);\n      resolvePromise = resolve;\n    });\n  };\n  \n  // Create enhanced render function\n  const enhancedRender = createEnhancedRender(tool.render, statusSignal, respond);\n  \n  // Create the frontend tool\n  const frontendTool: AngularFrontendTool<T> = {\n    ...tool,\n    handler,\n    render: enhancedRender\n  };\n  \n  // Add the tool (returns void, so we use the tool name as ID)\n  service.copilotkit.addTool(frontendTool);\n  const toolId = frontendTool.name;\n  \n  // Register tool render if provided\n  if (frontendTool.render && tool.parameters) {\n    service.registerToolRender(frontendTool.name, {\n      name: frontendTool.name,\n      args: tool.parameters,\n      render: frontendTool.render\n    });\n  }\n  \n  // Cleanup on destroy\n  destroyRef.onDestroy(() => {\n    service.copilotkit.removeTool(toolId);\n    if (frontendTool.render) {\n      service.unregisterToolRender(frontendTool.name);\n    }\n  });\n  \n  return toolId;\n}\n\n/**\n * Adds a human-in-the-loop tool with explicit service parameter.\n * Returns a cleanup function.\n * \n * @param service - The CopilotKitService instance\n * @param tool - The human-in-the-loop tool configuration\n * @returns Cleanup function to remove the tool\n * \n * @example\n * ```typescript\n * export class MyComponent implements OnInit, OnDestroy {\n *   private cleanup?: () => void;\n *   \n *   constructor(private copilotkit: CopilotKitService) {}\n *   \n *   ngOnInit() {\n *     this.cleanup = addHumanInTheLoop(this.copilotkit, {\n *       name: 'requireApproval',\n *       description: 'Requires user approval',\n *       args: z.object({ action: z.string() }),\n *       render: ApprovalDialogComponent\n *     });\n *   }\n *   \n *   ngOnDestroy() {\n *     this.cleanup?.();\n *   }\n * }\n * ```\n */\nexport function addHumanInTheLoop<T extends Record<string, any> = Record<string, any>>(\n  service: CopilotKitService,\n  tool: AngularHumanInTheLoop<T>\n): () => void {\n  // Create state management\n  const statusSignal = signal<ToolCallStatus>(ToolCallStatus.InProgress);\n  let resolvePromise: ((result: unknown) => void) | null = null;\n  \n  // Create respond function\n  const respond = async (result: unknown): Promise<void> => {\n    if (resolvePromise) {\n      resolvePromise(result);\n      statusSignal.set(ToolCallStatus.Complete);\n      resolvePromise = null;\n    }\n  };\n  \n  // Create handler that returns a Promise\n  const handler = async (_args: T): Promise<unknown> => {\n    return new Promise((resolve) => {\n      statusSignal.set(ToolCallStatus.Executing);\n      resolvePromise = resolve;\n    });\n  };\n  \n  // Create enhanced render function\n  const enhancedRender = createEnhancedRender(tool.render, statusSignal, respond);\n  \n  // Create the frontend tool\n  const frontendTool: AngularFrontendTool<T> = {\n    ...tool,\n    handler,\n    render: enhancedRender\n  };\n  \n  // Add the tool (returns void, so we use the tool name as ID)\n  service.copilotkit.addTool(frontendTool);\n  const toolId = frontendTool.name;\n  \n  // Register tool render if provided\n  if (frontendTool.render && tool.parameters) {\n    service.registerToolRender(frontendTool.name, {\n      name: frontendTool.name,\n      args: tool.parameters,\n      render: frontendTool.render\n    });\n  }\n  \n  // Return cleanup function\n  return () => {\n    service.copilotkit.removeTool(toolId);\n    if (frontendTool.render) {\n      service.unregisterToolRender(frontendTool.name);\n    }\n  };\n}\n\n/**\n * Creates a human-in-the-loop tool with dynamic update capabilities.\n * \n * @param service - The CopilotKitService instance\n * @param tool - The human-in-the-loop tool configuration\n * @returns Object with status signal, update and destroy methods\n * \n * @example\n * ```typescript\n * export class MyComponent {\n *   humanInTheLoop = createHumanInTheLoop(this.copilotkit, {\n *     name: 'requireApproval',\n *     description: 'Requires user approval',\n *     args: z.object({ action: z.string() }),\n *     render: ApprovalDialogComponent\n *   });\n *   \n *   updateDescription(newDesc: string) {\n *     this.humanInTheLoop.update({ description: newDesc });\n *   }\n *   \n *   ngOnDestroy() {\n *     this.humanInTheLoop.destroy();\n *   }\n * }\n * ```\n */\nexport function createHumanInTheLoop<T extends Record<string, any> = Record<string, any>>(\n  service: CopilotKitService,\n  tool: AngularHumanInTheLoop<T>\n): HumanInTheLoopState & { update: (updates: Partial<AngularHumanInTheLoop<T>>) => void } {\n  // Create state management\n  const statusSignal = signal<ToolCallStatus>(ToolCallStatus.InProgress);\n  let currentTool = { ...tool };\n  let toolId: string = '';\n  let resolvePromise: ((result: unknown) => void) | null = null;\n  \n  // Create respond function\n  const respond = async (result: unknown): Promise<void> => {\n    if (resolvePromise) {\n      resolvePromise(result);\n      statusSignal.set(ToolCallStatus.Complete);\n      resolvePromise = null;\n    }\n  };\n  \n  // Create handler that returns a Promise\n  const handler = async (_args: T): Promise<unknown> => {\n    return new Promise((resolve) => {\n      statusSignal.set(ToolCallStatus.Executing);\n      resolvePromise = resolve;\n    });\n  };\n  \n  // Function to add the tool\n  const addTool = () => {\n    // Create enhanced render function\n    const enhancedRender = createEnhancedRender(currentTool.render, statusSignal, respond);\n    \n    // Create the frontend tool\n    const frontendTool: AngularFrontendTool<T> = {\n      ...currentTool,\n      handler,\n      render: enhancedRender\n    };\n    \n    // Add tool (returns void, so we use the tool name as ID)\n    service.copilotkit.addTool(frontendTool);\n    toolId = frontendTool.name;\n    \n    // Register tool render if provided\n    if (frontendTool.render && currentTool.parameters) {\n      service.registerToolRender(frontendTool.name, {\n        name: frontendTool.name,\n        args: currentTool.parameters,\n        render: frontendTool.render\n      });\n    }\n  };\n  \n  // Initialize the tool\n  addTool();\n  \n  return {\n    status: statusSignal.asReadonly(),\n    toolId,\n    update: (updates: Partial<AngularHumanInTheLoop<T>>) => {\n      // Remove old tool\n      service.copilotkit.removeTool(toolId);\n      if (currentTool.render) {\n        service.unregisterToolRender(currentTool.name);\n      }\n      \n      // Update tool configuration\n      currentTool = { ...currentTool, ...updates };\n      \n      // Re-add with new configuration\n      addTool();\n    },\n    destroy: () => {\n      service.copilotkit.removeTool(toolId);\n      if (currentTool.render) {\n        service.unregisterToolRender(currentTool.name);\n      }\n    }\n  };\n}\n\n/**\n * Creates an enhanced render function that injects the respond function\n * when the status is 'executing'.\n */\nfunction createEnhancedRender<T extends Record<string, any>>(\n  originalRender: Type<any> | TemplateRef<HumanInTheLoopProps<T>>,\n  _statusSignal: Signal<ToolCallStatus>,\n  _respond: (result: unknown) => Promise<void>\n): Type<any> | TemplateRef<any> {\n  // For component classes, we need to create a wrapper\n  if (isComponentClass(originalRender)) {\n    // Return a wrapper component factory\n    // This is complex in Angular and would require dynamic component creation\n    // For now, we'll return the original and rely on prop injection\n    return originalRender;\n  }\n  \n  // For templates, we can't easily wrap them\n  // The template context will be enhanced in the render component\n  return originalRender;\n}\n\n/**\n * Helper function to check if a value is a component class\n */\nfunction isComponentClass(value: any): value is Type<any> {\n  return typeof value === 'function' && value.prototype;\n}\n\n/**\n * Enhanced component wrapper for human-in-the-loop.\n * This would be used internally by the tool render component to inject\n * the respond function based on status.\n * \n * @internal\n */\nexport function enhancePropsForHumanInTheLoop<T>(\n  props: HumanInTheLoopProps<T>,\n  status: ToolCallStatus,\n  respond?: (result: unknown) => Promise<void>\n): HumanInTheLoopProps<T> {\n  if (status === ToolCallStatus.Executing && respond) {\n    return {\n      ...props,\n      status: ToolCallStatus.Executing,\n      respond\n    } as HumanInTheLoopProps<T>;\n  }\n  \n  if (status === ToolCallStatus.Complete) {\n    return {\n      ...props,\n      status: ToolCallStatus.Complete,\n      result: typeof props.result === 'string' ? props.result : '',\n      respond: undefined\n    } as HumanInTheLoopProps<T>;\n  }\n  \n  // InProgress\n  return {\n    ...props,\n    status: ToolCallStatus.InProgress,\n    result: undefined,\n    respond: undefined\n  } as HumanInTheLoopProps<T>;\n}"]}
|
|
293
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"human-in-the-loop.utils.js","sourceRoot":"","sources":["../../../src/utils/human-in-the-loop.utils.ts"],"names":[],"mappings":"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","sourcesContent":["import { \n  DestroyRef, \n  inject, \n  signal, \n  Signal,\n  Type,\n  TemplateRef\n} from '@angular/core';\nimport { CopilotKitService } from '../core/copilotkit.service';\nimport { \n  AngularHumanInTheLoop, \n  ToolCallStatus,\n  HumanInTheLoopState,\n  HumanInTheLoopProps,\n  AngularFrontendTool\n} from '../core/copilotkit.types';\n\n/**\n * Registers a human-in-the-loop tool that requires user interaction.\n * Must be called within an injection context.\n * Automatically cleans up when the component/service is destroyed.\n * \n * @param tool - The human-in-the-loop tool configuration\n * @returns The tool ID\n * \n * @example\n * ```typescript\n * export class ApprovalComponent {\n *   toolId = registerHumanInTheLoop({\n *     name: 'requireApproval',\n *     description: 'Requires user approval',\n *     args: z.object({ action: z.string() }),\n *     render: ApprovalDialogComponent\n *   });\n * }\n * ```\n */\nexport function registerHumanInTheLoop<T extends Record<string, any> = Record<string, any>>(\n  tool: AngularHumanInTheLoop<T>\n): string {\n  const service = inject(CopilotKitService);\n  const destroyRef = inject(DestroyRef);\n  \n  // Create state management\n  const statusSignal = signal<ToolCallStatus>(ToolCallStatus.InProgress);\n  let resolvePromise: ((result: unknown) => void) | null = null;\n  \n  // Create respond function\n  const respond = async (result: unknown): Promise<void> => {\n    if (resolvePromise) {\n      resolvePromise(result);\n      statusSignal.set(ToolCallStatus.Complete);\n      resolvePromise = null;\n    }\n  };\n  \n  // Create handler that returns a Promise\n  const handler = async (_args: T): Promise<unknown> => {\n    return new Promise((resolve) => {\n      statusSignal.set(ToolCallStatus.Executing);\n      resolvePromise = resolve;\n    });\n  };\n  \n  // Create enhanced render function\n  const enhancedRender = createEnhancedRender(tool.render, statusSignal, respond);\n  \n  // Create the frontend tool\n  const frontendTool: AngularFrontendTool<T> = {\n    ...tool,\n    handler,\n    render: enhancedRender\n  };\n  \n  // Add the tool (returns void, so we use the tool name as ID)\n  service.copilotkit.addTool(frontendTool);\n  const toolId = frontendTool.name;\n  \n  // Register tool render if provided\n  if (frontendTool.render) {\n    service.registerToolRender(frontendTool.name, {\n      name: frontendTool.name,\n      render: frontendTool.render\n    });\n  }\n  \n  // Cleanup on destroy\n  destroyRef.onDestroy(() => {\n    service.copilotkit.removeTool(toolId);\n    if (frontendTool.render) {\n      service.unregisterToolRender(frontendTool.name);\n    }\n  });\n  \n  return toolId;\n}\n\n/**\n * Adds a human-in-the-loop tool with explicit service parameter.\n * Returns a cleanup function.\n * \n * @param service - The CopilotKitService instance\n * @param tool - The human-in-the-loop tool configuration\n * @returns Cleanup function to remove the tool\n * \n * @example\n * ```typescript\n * export class MyComponent implements OnInit, OnDestroy {\n *   private cleanup?: () => void;\n *   \n *   constructor(private copilotkit: CopilotKitService) {}\n *   \n *   ngOnInit() {\n *     this.cleanup = addHumanInTheLoop(this.copilotkit, {\n *       name: 'requireApproval',\n *       description: 'Requires user approval',\n *       args: z.object({ action: z.string() }),\n *       render: ApprovalDialogComponent\n *     });\n *   }\n *   \n *   ngOnDestroy() {\n *     this.cleanup?.();\n *   }\n * }\n * ```\n */\nexport function addHumanInTheLoop<T extends Record<string, any> = Record<string, any>>(\n  service: CopilotKitService,\n  tool: AngularHumanInTheLoop<T>\n): () => void {\n  // Create state management\n  const statusSignal = signal<ToolCallStatus>(ToolCallStatus.InProgress);\n  let resolvePromise: ((result: unknown) => void) | null = null;\n  \n  // Create respond function\n  const respond = async (result: unknown): Promise<void> => {\n    if (resolvePromise) {\n      resolvePromise(result);\n      statusSignal.set(ToolCallStatus.Complete);\n      resolvePromise = null;\n    }\n  };\n  \n  // Create handler that returns a Promise\n  const handler = async (_args: T): Promise<unknown> => {\n    return new Promise((resolve) => {\n      statusSignal.set(ToolCallStatus.Executing);\n      resolvePromise = resolve;\n    });\n  };\n  \n  // Create enhanced render function\n  const enhancedRender = createEnhancedRender(tool.render, statusSignal, respond);\n  \n  // Create the frontend tool\n  const frontendTool: AngularFrontendTool<T> = {\n    ...tool,\n    handler,\n    render: enhancedRender\n  };\n  \n  // Add the tool (returns void, so we use the tool name as ID)\n  service.copilotkit.addTool(frontendTool);\n  const toolId = frontendTool.name;\n  \n  // Register tool render if provided\n  if (frontendTool.render) {\n    service.registerToolRender(frontendTool.name, {\n      name: frontendTool.name,\n      render: frontendTool.render\n    });\n  }\n  \n  // Return cleanup function\n  return () => {\n    service.copilotkit.removeTool(toolId);\n    if (frontendTool.render) {\n      service.unregisterToolRender(frontendTool.name);\n    }\n  };\n}\n\n/**\n * Creates a human-in-the-loop tool with dynamic update capabilities.\n * \n * @param service - The CopilotKitService instance\n * @param tool - The human-in-the-loop tool configuration\n * @returns Object with status signal, update and destroy methods\n * \n * @example\n * ```typescript\n * export class MyComponent {\n *   humanInTheLoop = createHumanInTheLoop(this.copilotkit, {\n *     name: 'requireApproval',\n *     description: 'Requires user approval',\n *     args: z.object({ action: z.string() }),\n *     render: ApprovalDialogComponent\n *   });\n *   \n *   updateDescription(newDesc: string) {\n *     this.humanInTheLoop.update({ description: newDesc });\n *   }\n *   \n *   ngOnDestroy() {\n *     this.humanInTheLoop.destroy();\n *   }\n * }\n * ```\n */\nexport function createHumanInTheLoop<T extends Record<string, any> = Record<string, any>>(\n  service: CopilotKitService,\n  tool: AngularHumanInTheLoop<T>\n): HumanInTheLoopState & { update: (updates: Partial<AngularHumanInTheLoop<T>>) => void } {\n  // Create state management\n  const statusSignal = signal<ToolCallStatus>(ToolCallStatus.InProgress);\n  let currentTool = { ...tool };\n  let toolId: string = '';\n  let resolvePromise: ((result: unknown) => void) | null = null;\n  \n  // Create respond function\n  const respond = async (result: unknown): Promise<void> => {\n    if (resolvePromise) {\n      resolvePromise(result);\n      statusSignal.set(ToolCallStatus.Complete);\n      resolvePromise = null;\n    }\n  };\n  \n  // Create handler that returns a Promise\n  const handler = async (_args: T): Promise<unknown> => {\n    return new Promise((resolve) => {\n      statusSignal.set(ToolCallStatus.Executing);\n      resolvePromise = resolve;\n    });\n  };\n  \n  // Function to add the tool\n  const addTool = () => {\n    // Create enhanced render function\n    const enhancedRender = createEnhancedRender(currentTool.render, statusSignal, respond);\n    \n    // Create the frontend tool\n    const frontendTool: AngularFrontendTool<T> = {\n      ...currentTool,\n      handler,\n      render: enhancedRender\n    };\n    \n    // Add tool (returns void, so we use the tool name as ID)\n    service.copilotkit.addTool(frontendTool);\n    toolId = frontendTool.name;\n    \n    // Register tool render if provided\n    if (frontendTool.render) {\n      service.registerToolRender(frontendTool.name, {\n        name: frontendTool.name,\n        render: frontendTool.render\n      });\n    }\n  };\n  \n  // Initialize the tool\n  addTool();\n  \n  return {\n    status: statusSignal.asReadonly(),\n    toolId,\n    update: (updates: Partial<AngularHumanInTheLoop<T>>) => {\n      // Remove old tool\n      service.copilotkit.removeTool(toolId);\n      if (currentTool.render) {\n        service.unregisterToolRender(currentTool.name);\n      }\n      \n      // Update tool configuration\n      currentTool = { ...currentTool, ...updates };\n      \n      // Re-add with new configuration\n      addTool();\n    },\n    destroy: () => {\n      service.copilotkit.removeTool(toolId);\n      if (currentTool.render) {\n        service.unregisterToolRender(currentTool.name);\n      }\n    }\n  };\n}\n\n/**\n * Creates an enhanced render function that injects the respond function\n * when the status is 'executing'.\n */\nfunction createEnhancedRender<T extends Record<string, any>>(\n  originalRender: Type<any> | TemplateRef<HumanInTheLoopProps<T>>,\n  _statusSignal: Signal<ToolCallStatus>,\n  _respond: (result: unknown) => Promise<void>\n): Type<any> | TemplateRef<any> {\n  // For component classes, we need to create a wrapper\n  if (isComponentClass(originalRender)) {\n    // Return a wrapper component factory\n    // This is complex in Angular and would require dynamic component creation\n    // For now, we'll return the original and rely on prop injection\n    return originalRender;\n  }\n  \n  // For templates, we can't easily wrap them\n  // The template context will be enhanced in the render component\n  return originalRender;\n}\n\n/**\n * Helper function to check if a value is a component class\n */\nfunction isComponentClass(value: any): value is Type<any> {\n  return typeof value === 'function' && value.prototype;\n}\n\n/**\n * Enhanced component wrapper for human-in-the-loop.\n * This would be used internally by the tool render component to inject\n * the respond function based on status.\n * \n * @internal\n */\nexport function enhancePropsForHumanInTheLoop<T>(\n  props: HumanInTheLoopProps<T>,\n  status: ToolCallStatus,\n  respond?: (result: unknown) => Promise<void>\n): HumanInTheLoopProps<T> {\n  if (status === ToolCallStatus.Executing && respond) {\n    return {\n      ...props,\n      status: ToolCallStatus.Executing,\n      respond\n    } as HumanInTheLoopProps<T>;\n  }\n  \n  if (status === ToolCallStatus.Complete) {\n    return {\n      ...props,\n      status: ToolCallStatus.Complete,\n      result: typeof props.result === 'string' ? props.result : '',\n      respond: undefined\n    } as HumanInTheLoopProps<T>;\n  }\n  \n  // InProgress\n  return {\n    ...props,\n    status: ToolCallStatus.InProgress,\n    result: undefined,\n    respond: undefined\n  } as HumanInTheLoopProps<T>;\n}"]}
|
|
@@ -3,7 +3,6 @@ import { InjectionToken, signal, computed, effect, untracked, Inject, Injectable
|
|
|
3
3
|
import { toObservable } from '@angular/core/rxjs-interop';
|
|
4
4
|
import { CopilotKitCore, ToolCallStatus, completePartialMarkdown } from '@copilotkitnext/core';
|
|
5
5
|
export { ToolCallStatus } from '@copilotkitnext/core';
|
|
6
|
-
import { z } from 'zod';
|
|
7
6
|
import { DEFAULT_AGENT_ID, partialJSONParse, randomUUID } from '@copilotkitnext/shared';
|
|
8
7
|
import { Overlay, OverlayPositionBuilder, OverlayModule } from '@angular/cdk/overlay';
|
|
9
8
|
import { ComponentPortal } from '@angular/cdk/portal';
|
|
@@ -23,6 +22,7 @@ import { ScrollingModule } from '@angular/cdk/scrolling';
|
|
|
23
22
|
import { Subject, BehaviorSubject, merge, fromEvent, animationFrameScheduler } from 'rxjs';
|
|
24
23
|
import { startWith, throttleTime, map, distinctUntilChanged, takeUntil, debounceTime, filter } from 'rxjs/operators';
|
|
25
24
|
|
|
25
|
+
1;
|
|
26
26
|
// Injection tokens for dependency injection
|
|
27
27
|
const COPILOTKIT_RUNTIME_URL = new InjectionToken("COPILOTKIT_RUNTIME_URL");
|
|
28
28
|
const COPILOTKIT_HEADERS = new InjectionToken("COPILOTKIT_HEADERS", { factory: () => ({}) });
|
|
@@ -133,10 +133,9 @@ class CopilotKitService {
|
|
|
133
133
|
const combined = [...this._renderToolCalls()];
|
|
134
134
|
// Add render components from frontend tools
|
|
135
135
|
this._frontendTools().forEach((tool) => {
|
|
136
|
-
if (tool.render
|
|
136
|
+
if (tool.render) {
|
|
137
137
|
combined.push({
|
|
138
138
|
name: tool.name,
|
|
139
|
-
args: tool.parameters,
|
|
140
139
|
render: tool.render,
|
|
141
140
|
...(tool.agentId && { agentId: tool.agentId }),
|
|
142
141
|
});
|
|
@@ -144,10 +143,9 @@ class CopilotKitService {
|
|
|
144
143
|
});
|
|
145
144
|
// Add render components from human-in-the-loop tools
|
|
146
145
|
this._humanInTheLoop().forEach((tool) => {
|
|
147
|
-
if (tool.render
|
|
146
|
+
if (tool.render) {
|
|
148
147
|
combined.push({
|
|
149
148
|
name: tool.name,
|
|
150
|
-
args: tool.parameters,
|
|
151
149
|
render: tool.render,
|
|
152
150
|
...(tool.agentId && { agentId: tool.agentId }),
|
|
153
151
|
});
|
|
@@ -202,10 +200,9 @@ class CopilotKitService {
|
|
|
202
200
|
frontendTools.forEach((tool) => {
|
|
203
201
|
allTools[tool.name] = tool;
|
|
204
202
|
// Add render component if provided
|
|
205
|
-
if (tool.render
|
|
203
|
+
if (tool.render) {
|
|
206
204
|
allRenderToolCalls.push({
|
|
207
205
|
name: tool.name,
|
|
208
|
-
args: tool.parameters,
|
|
209
206
|
render: tool.render,
|
|
210
207
|
...(tool.agentId && { agentId: tool.agentId }),
|
|
211
208
|
});
|
|
@@ -228,10 +225,9 @@ class CopilotKitService {
|
|
|
228
225
|
};
|
|
229
226
|
allTools[tool.name] = frontendTool;
|
|
230
227
|
// Add the render component
|
|
231
|
-
if (tool.render
|
|
228
|
+
if (tool.render) {
|
|
232
229
|
allRenderToolCalls.push({
|
|
233
230
|
name: tool.name,
|
|
234
|
-
args: tool.parameters,
|
|
235
231
|
render: tool.render,
|
|
236
232
|
...(tool.agentId && { agentId: tool.agentId }),
|
|
237
233
|
});
|
|
@@ -263,16 +259,7 @@ class CopilotKitService {
|
|
|
263
259
|
});
|
|
264
260
|
}
|
|
265
261
|
});
|
|
266
|
-
//
|
|
267
|
-
effect(() => {
|
|
268
|
-
const current = this._renderToolCalls();
|
|
269
|
-
if (current !== this.initialRenderToolCalls &&
|
|
270
|
-
this.initialRenderToolCalls.length > 0) {
|
|
271
|
-
untracked(() => {
|
|
272
|
-
console.error("renderToolCalls must be a stable object. To add/remove tools dynamically, use dynamic tool registration.");
|
|
273
|
-
});
|
|
274
|
-
}
|
|
275
|
-
});
|
|
262
|
+
// Previously warned if renderToolCalls reference changed; removed per UX feedback
|
|
276
263
|
}
|
|
277
264
|
/**
|
|
278
265
|
* Setup effects to sync runtime configuration with CopilotKitCore
|
|
@@ -370,9 +357,6 @@ class CopilotKitService {
|
|
|
370
357
|
* Update render tool calls (warns if object reference changes)
|
|
371
358
|
*/
|
|
372
359
|
setRenderToolCalls(renderToolCalls) {
|
|
373
|
-
if (renderToolCalls !== this.initialRenderToolCalls) {
|
|
374
|
-
console.error("renderToolCalls must be a stable object. To add/remove tools dynamically, use dynamic tool registration.");
|
|
375
|
-
}
|
|
376
360
|
this._renderToolCalls.set(renderToolCalls);
|
|
377
361
|
}
|
|
378
362
|
/**
|
|
@@ -899,7 +883,6 @@ function addFrontendTool(service, tool) {
|
|
|
899
883
|
else {
|
|
900
884
|
const renderEntry = {
|
|
901
885
|
name: tool.name,
|
|
902
|
-
args: tool.parameters || z.object({}),
|
|
903
886
|
render: tool.render
|
|
904
887
|
};
|
|
905
888
|
service.setCurrentRenderToolCalls([...currentRenders, renderEntry]);
|
|
@@ -952,7 +935,6 @@ function registerFrontendTool(tool) {
|
|
|
952
935
|
else {
|
|
953
936
|
const renderEntry = {
|
|
954
937
|
name: tool.name,
|
|
955
|
-
args: tool.parameters || z.object({}),
|
|
956
938
|
render: tool.render
|
|
957
939
|
};
|
|
958
940
|
service.setCurrentRenderToolCalls([...currentRenders, renderEntry]);
|
|
@@ -1047,7 +1029,6 @@ function createDynamicFrontendTool(name, description, parameters, handler, rende
|
|
|
1047
1029
|
const currentRenders = service.currentRenderToolCalls();
|
|
1048
1030
|
const renderEntry = {
|
|
1049
1031
|
name: name,
|
|
1050
|
-
args: parameters,
|
|
1051
1032
|
render: currentRender
|
|
1052
1033
|
};
|
|
1053
1034
|
const existingIndex = currentRenders.findIndex((r) => r.name === name);
|
|
@@ -1328,10 +1309,9 @@ function registerHumanInTheLoop(tool) {
|
|
|
1328
1309
|
service.copilotkit.addTool(frontendTool);
|
|
1329
1310
|
const toolId = frontendTool.name;
|
|
1330
1311
|
// Register tool render if provided
|
|
1331
|
-
if (frontendTool.render
|
|
1312
|
+
if (frontendTool.render) {
|
|
1332
1313
|
service.registerToolRender(frontendTool.name, {
|
|
1333
1314
|
name: frontendTool.name,
|
|
1334
|
-
args: tool.parameters,
|
|
1335
1315
|
render: frontendTool.render
|
|
1336
1316
|
});
|
|
1337
1317
|
}
|
|
@@ -1405,10 +1385,9 @@ function addHumanInTheLoop(service, tool) {
|
|
|
1405
1385
|
service.copilotkit.addTool(frontendTool);
|
|
1406
1386
|
const toolId = frontendTool.name;
|
|
1407
1387
|
// Register tool render if provided
|
|
1408
|
-
if (frontendTool.render
|
|
1388
|
+
if (frontendTool.render) {
|
|
1409
1389
|
service.registerToolRender(frontendTool.name, {
|
|
1410
1390
|
name: frontendTool.name,
|
|
1411
|
-
args: tool.parameters,
|
|
1412
1391
|
render: frontendTool.render
|
|
1413
1392
|
});
|
|
1414
1393
|
}
|
|
@@ -1482,10 +1461,9 @@ function createHumanInTheLoop(service, tool) {
|
|
|
1482
1461
|
service.copilotkit.addTool(frontendTool);
|
|
1483
1462
|
toolId = frontendTool.name;
|
|
1484
1463
|
// Register tool render if provided
|
|
1485
|
-
if (frontendTool.render
|
|
1464
|
+
if (frontendTool.render) {
|
|
1486
1465
|
service.registerToolRender(frontendTool.name, {
|
|
1487
1466
|
name: frontendTool.name,
|
|
1488
|
-
args: currentTool.parameters,
|
|
1489
1467
|
render: frontendTool.render
|
|
1490
1468
|
});
|
|
1491
1469
|
}
|
|
@@ -2218,7 +2196,6 @@ class CopilotKitFrontendToolDirective {
|
|
|
2218
2196
|
const currentRenders = this.copilotkit.currentRenderToolCalls();
|
|
2219
2197
|
const renderEntry = {
|
|
2220
2198
|
name: tool.name,
|
|
2221
|
-
args: tool.parameters || z.object({}),
|
|
2222
2199
|
render: tool.render,
|
|
2223
2200
|
};
|
|
2224
2201
|
// Check for duplicate
|
|
@@ -2674,7 +2651,6 @@ class CopilotKitHumanInTheLoopDirective {
|
|
|
2674
2651
|
// Register the render with respond capability
|
|
2675
2652
|
this.copilotkit.registerToolRender(this.name, {
|
|
2676
2653
|
name: this.name,
|
|
2677
|
-
args: this.parameters,
|
|
2678
2654
|
render: this.createEnhancedRender(),
|
|
2679
2655
|
});
|
|
2680
2656
|
}
|
|
@@ -3051,20 +3027,17 @@ class CopilotKitToolRenderComponent {
|
|
|
3051
3027
|
this.container.clear();
|
|
3052
3028
|
// Create the component
|
|
3053
3029
|
this.componentRef = this.container.createComponent(componentClass);
|
|
3054
|
-
//
|
|
3055
|
-
|
|
3056
|
-
|
|
3030
|
+
// Determine declared inputs to avoid NG0303 logs
|
|
3031
|
+
const cmpDef = componentClass.ɵcmp;
|
|
3032
|
+
const declaredInputs = new Set(Object.keys(cmpDef?.inputs ?? {}));
|
|
3033
|
+
if (declaredInputs.has('props')) {
|
|
3057
3034
|
this.componentRef.setInput('props', props);
|
|
3058
3035
|
}
|
|
3059
|
-
|
|
3060
|
-
// If props input doesn't exist, try setting individual inputs
|
|
3036
|
+
else {
|
|
3061
3037
|
for (const [key, value] of Object.entries(props)) {
|
|
3062
|
-
|
|
3038
|
+
if (declaredInputs.has(key)) {
|
|
3063
3039
|
this.componentRef.setInput(key, value);
|
|
3064
3040
|
}
|
|
3065
|
-
catch (inputError) {
|
|
3066
|
-
// Input might not exist on the component, skip it
|
|
3067
|
-
}
|
|
3068
3041
|
}
|
|
3069
3042
|
}
|
|
3070
3043
|
// Trigger change detection
|
|
@@ -3204,10 +3177,19 @@ function createComponent(viewContainer, component, props, injector, outputs) {
|
|
|
3204
3177
|
injector
|
|
3205
3178
|
});
|
|
3206
3179
|
if (props) {
|
|
3207
|
-
// Apply props using setInput
|
|
3208
|
-
|
|
3209
|
-
|
|
3210
|
-
|
|
3180
|
+
// Apply props using setInput, but only for declared inputs
|
|
3181
|
+
const cmpDef = component.ɵcmp;
|
|
3182
|
+
const declaredInputs = new Set(Object.keys(cmpDef?.inputs ?? {}));
|
|
3183
|
+
if (declaredInputs.has('props')) {
|
|
3184
|
+
componentRef.setInput('props', props);
|
|
3185
|
+
}
|
|
3186
|
+
else {
|
|
3187
|
+
for (const key in props) {
|
|
3188
|
+
if (declaredInputs.has(key)) {
|
|
3189
|
+
const value = props[key];
|
|
3190
|
+
componentRef.setInput(key, value);
|
|
3191
|
+
}
|
|
3192
|
+
}
|
|
3211
3193
|
}
|
|
3212
3194
|
}
|
|
3213
3195
|
if (outputs) {
|
|
@@ -3442,11 +3424,21 @@ class CopilotSlotComponent {
|
|
|
3442
3424
|
return;
|
|
3443
3425
|
}
|
|
3444
3426
|
const props = this.context;
|
|
3445
|
-
// Update props using setInput
|
|
3427
|
+
// Update props using setInput, only for declared inputs
|
|
3446
3428
|
if (props) {
|
|
3447
|
-
|
|
3448
|
-
|
|
3449
|
-
|
|
3429
|
+
const ctor = this.componentRef.instance.constructor;
|
|
3430
|
+
const cmpDef = ctor?.ɵcmp;
|
|
3431
|
+
const declaredInputs = new Set(Object.keys(cmpDef?.inputs ?? {}));
|
|
3432
|
+
if (declaredInputs.has('props')) {
|
|
3433
|
+
this.componentRef.setInput('props', props);
|
|
3434
|
+
}
|
|
3435
|
+
else {
|
|
3436
|
+
for (const key in props) {
|
|
3437
|
+
if (declaredInputs.has(key)) {
|
|
3438
|
+
const value = props[key];
|
|
3439
|
+
this.componentRef.setInput(key, value);
|
|
3440
|
+
}
|
|
3441
|
+
}
|
|
3450
3442
|
}
|
|
3451
3443
|
}
|
|
3452
3444
|
// Trigger change detection
|
|
@@ -3571,7 +3563,7 @@ class CopilotChatTextareaComponent {
|
|
|
3571
3563
|
if (configValue !== undefined && !this.customPlaceholder()) {
|
|
3572
3564
|
this.value.set(configValue);
|
|
3573
3565
|
}
|
|
3574
|
-
});
|
|
3566
|
+
}, { allowSignalWrites: true });
|
|
3575
3567
|
}
|
|
3576
3568
|
ngAfterViewInit() {
|
|
3577
3569
|
this.calculateMaxHeight();
|
|
@@ -4574,7 +4566,7 @@ class CopilotChatInputComponent {
|
|
|
4574
4566
|
if (configValue !== undefined && !this.valueSignal()) {
|
|
4575
4567
|
this.valueSignal.set(configValue);
|
|
4576
4568
|
}
|
|
4577
|
-
});
|
|
4569
|
+
}, { allowSignalWrites: true });
|
|
4578
4570
|
}
|
|
4579
4571
|
// Output maps for slots
|
|
4580
4572
|
addFileButtonOutputs = { clicked: () => this.handleAddFile() };
|
|
@@ -5952,13 +5944,19 @@ class CopilotChatToolCallsViewComponent {
|
|
|
5952
5944
|
// Create the component
|
|
5953
5945
|
const componentRef = this.container.createComponent(componentClass);
|
|
5954
5946
|
this.componentRefs.set(toolCallId, componentRef);
|
|
5955
|
-
//
|
|
5956
|
-
|
|
5957
|
-
|
|
5958
|
-
|
|
5959
|
-
|
|
5960
|
-
|
|
5961
|
-
|
|
5947
|
+
// Determine declared inputs to avoid Angular NG0303 console errors
|
|
5948
|
+
const cmpDef = componentClass.ɵcmp;
|
|
5949
|
+
const declaredInputs = new Set(Object.keys(cmpDef?.inputs ?? {}));
|
|
5950
|
+
// Prefer a single 'props' input if declared
|
|
5951
|
+
if (declaredInputs.has('props')) {
|
|
5952
|
+
componentRef.setInput('props', props);
|
|
5953
|
+
}
|
|
5954
|
+
else {
|
|
5955
|
+
// Otherwise, set only inputs that the component actually declares
|
|
5956
|
+
for (const [key, value] of Object.entries(props)) {
|
|
5957
|
+
if (declaredInputs.has(key)) {
|
|
5958
|
+
componentRef.setInput(key, value);
|
|
5959
|
+
}
|
|
5962
5960
|
}
|
|
5963
5961
|
}
|
|
5964
5962
|
// Trigger change detection
|
|
@@ -6802,6 +6800,12 @@ class CopilotChatAssistantMessageComponent {
|
|
|
6802
6800
|
toolbarContext = computed(() => ({
|
|
6803
6801
|
children: null // Will be populated by the toolbar content
|
|
6804
6802
|
}));
|
|
6803
|
+
// Return true if assistant message has non-empty text content
|
|
6804
|
+
hasMessageContent() {
|
|
6805
|
+
const raw = (this.message?.content ?? '');
|
|
6806
|
+
const content = typeof raw === 'string' ? raw : String(raw ?? '');
|
|
6807
|
+
return content.trim().length > 0;
|
|
6808
|
+
}
|
|
6805
6809
|
toolCallsViewContext = computed(() => ({
|
|
6806
6810
|
message: this.message,
|
|
6807
6811
|
messages: this.messages,
|
|
@@ -6859,8 +6863,8 @@ class CopilotChatAssistantMessageComponent {
|
|
|
6859
6863
|
</copilot-chat-tool-calls-view>
|
|
6860
6864
|
}
|
|
6861
6865
|
|
|
6862
|
-
<!-- Toolbar -->
|
|
6863
|
-
<ng-container *ngIf="toolbarVisible">
|
|
6866
|
+
<!-- Toolbar: show only when there is assistant text content -->
|
|
6867
|
+
<ng-container *ngIf="toolbarVisible && hasMessageContent()">
|
|
6864
6868
|
@if (toolbarTemplate || toolbarComponent) {
|
|
6865
6869
|
<copilot-slot
|
|
6866
6870
|
[slot]="toolbarTemplate || toolbarComponent"
|
|
@@ -6987,8 +6991,8 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
|
6987
6991
|
</copilot-chat-tool-calls-view>
|
|
6988
6992
|
}
|
|
6989
6993
|
|
|
6990
|
-
<!-- Toolbar -->
|
|
6991
|
-
<ng-container *ngIf="toolbarVisible">
|
|
6994
|
+
<!-- Toolbar: show only when there is assistant text content -->
|
|
6995
|
+
<ng-container *ngIf="toolbarVisible && hasMessageContent()">
|
|
6992
6996
|
@if (toolbarTemplate || toolbarComponent) {
|
|
6993
6997
|
<copilot-slot
|
|
6994
6998
|
[slot]="toolbarTemplate || toolbarComponent"
|
|
@@ -9128,9 +9132,16 @@ class CopilotChatComponent {
|
|
|
9128
9132
|
a.threadId = this.threadId || this.generatedThreadId;
|
|
9129
9133
|
if (!this.hasConnectedOnce) {
|
|
9130
9134
|
this.hasConnectedOnce = true;
|
|
9131
|
-
|
|
9135
|
+
if ('isCopilotKitAgent' in a) {
|
|
9136
|
+
this.connectToAgent(a);
|
|
9137
|
+
}
|
|
9138
|
+
else {
|
|
9139
|
+
// Not a CopilotKit agent: ensure UI not showing loading cursor
|
|
9140
|
+
this.showCursor.set(false);
|
|
9141
|
+
this.cdr.markForCheck();
|
|
9142
|
+
}
|
|
9132
9143
|
}
|
|
9133
|
-
});
|
|
9144
|
+
}, { allowSignalWrites: true });
|
|
9134
9145
|
}
|
|
9135
9146
|
// Signals from watchAgent - using direct references instead of assignment
|
|
9136
9147
|
agent = signal(undefined).asReadonly();
|