@copilotkitnext/angular 0.0.5 → 0.0.7

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.
@@ -57,10 +57,19 @@ function createComponent(viewContainer, component, props, injector, outputs) {
57
57
  injector
58
58
  });
59
59
  if (props) {
60
- // Apply props using setInput
61
- for (const key in props) {
62
- const value = props[key];
63
- componentRef.setInput(key, value);
60
+ // Apply props using setInput, but only for declared inputs
61
+ const cmpDef = component.ɵcmp;
62
+ const declaredInputs = new Set(Object.keys(cmpDef?.inputs ?? {}));
63
+ if (declaredInputs.has('props')) {
64
+ componentRef.setInput('props', props);
65
+ }
66
+ else {
67
+ for (const key in props) {
68
+ if (declaredInputs.has(key)) {
69
+ const value = props[key];
70
+ componentRef.setInput(key, value);
71
+ }
72
+ }
64
73
  }
65
74
  }
66
75
  if (outputs) {
@@ -219,4 +228,4 @@ export function createSlotRenderer(defaultComponent, slotName) {
219
228
  });
220
229
  };
221
230
  }
222
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"slot.utils.js","sourceRoot":"","sources":["../../../../src/lib/slots/slot.utils.ts"],"names":[],"mappings":"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","sourcesContent":["import { \n  Type, \n  TemplateRef, \n  ViewContainerRef,\n  ComponentRef,\n  EmbeddedViewRef,\n  Injector,\n  inject\n} from '@angular/core';\nimport { SlotValue, RenderSlotOptions, SlotRegistryEntry, SLOT_CONFIG } from './slot.types';\n\n/**\n * Renders a slot value into a ViewContainerRef.\n * This is the core utility for slot rendering.\n * \n * @param viewContainer - The ViewContainerRef to render into\n * @param options - Options for rendering the slot\n * @returns The created component or embedded view reference\n * \n * @example\n * ```typescript\n * export class MyComponent {\n *   @ViewChild('container', { read: ViewContainerRef }) container!: ViewContainerRef;\n *   \n *   renderButton() {\n *     renderSlot(this.container, {\n *       slot: this.buttonOverride,\n *       defaultComponent: DefaultButton,\n *       props: { text: 'Click me' },\n *       outputs: { click: (event) => this.handleClick(event) }\n *     });\n *   }\n * }\n * ```\n */\nexport function renderSlot<T = any>(\n  viewContainer: ViewContainerRef,\n  options: RenderSlotOptions<T>\n): ComponentRef<T> | EmbeddedViewRef<T> | null {\n  const { slot, defaultComponent, props, injector, outputs } = options;\n  \n  viewContainer.clear();\n  \n  const effectiveSlot = slot ?? defaultComponent;\n  const effectiveInjector = injector ?? viewContainer.injector;\n  \n  if (effectiveSlot instanceof TemplateRef) {\n    // TemplateRef: render template\n    return viewContainer.createEmbeddedView(effectiveSlot, {\n      $implicit: props ?? {},\n      props: props ?? {}\n    } as any);\n  } else if (isComponentType(effectiveSlot)) {\n    // Component type - wrap in try/catch for safety\n    try {\n      return createComponent(\n        viewContainer,\n        effectiveSlot as Type<T>,\n        props,\n        effectiveInjector,\n        outputs\n      );\n    } catch (error) {\n      console.warn('Failed to create component:', effectiveSlot, error);\n      // Fall through to default component\n    }\n  }\n  \n  // Default: render default component if provided\n  return defaultComponent ? createComponent(\n    viewContainer,\n    defaultComponent,\n    props,\n    effectiveInjector,\n    outputs\n  ) : null;\n}\n\n/**\n * Creates a component and applies properties.\n */\nfunction createComponent<T>(\n  viewContainer: ViewContainerRef,\n  component: Type<T>,\n  props?: Partial<T>,\n  injector?: Injector,\n  outputs?: Record<string, (event: any) => void>\n): ComponentRef<T> {\n  const componentRef = viewContainer.createComponent(component, {\n    injector\n  });\n  \n  if (props) {\n    // Apply props using setInput\n    for (const key in props) {\n      const value = props[key];\n      componentRef.setInput(key, value);\n    }\n  }\n  \n  if (outputs) {\n    // Wire up output event handlers with proper cleanup\n    const instance = componentRef.instance as any;\n    const subscriptions: any[] = [];\n    \n    for (const [eventName, handler] of Object.entries(outputs)) {\n      if (instance[eventName]?.subscribe) {\n        const subscription = instance[eventName].subscribe(handler);\n        subscriptions.push(subscription);\n      }\n    }\n    \n    // Register cleanup on component destroy\n    componentRef.onDestroy(() => {\n      subscriptions.forEach(sub => sub.unsubscribe());\n    });\n  }\n  \n  // Trigger change detection\n  componentRef.changeDetectorRef.detectChanges();\n  \n  return componentRef;\n}\n\n\n/**\n * Checks if a value is a component type.\n * Simplified check - rely on try/catch for actual validation.\n */\nexport function isComponentType(value: any): boolean {\n  // Arrow functions and regular functions without a prototype are not components\n  return typeof value === 'function' && !!value.prototype;\n}\n\n/**\n * Checks if a value is a valid slot value.\n */\nexport function isSlotValue(value: any): value is SlotValue {\n  return value instanceof TemplateRef || isComponentType(value);\n}\n\n/**\n * Normalizes a slot value to a consistent format.\n */\nexport function normalizeSlotValue<T = any>(\n  value: SlotValue<T> | undefined,\n  defaultComponent: Type<T> | undefined\n): SlotRegistryEntry<T> {\n  if (!value) {\n    return { component: defaultComponent };\n  }\n  \n  if (value instanceof TemplateRef) {\n    return { template: value };\n  }\n  \n  if (isComponentType(value)) {\n    return { component: value as Type<T> };\n  }\n  \n  return { component: defaultComponent };\n}\n\n/**\n * Creates a slot configuration map for a component.\n * \n * @example\n * ```typescript\n * const slots = createSlotConfig({\n *   sendButton: CustomSendButton,\n *   toolbar: 'custom-toolbar-class',\n *   footer: footerTemplate\n * }, {\n *   sendButton: DefaultSendButton,\n *   toolbar: DefaultToolbar,\n *   footer: DefaultFooter\n * });\n * ```\n */\nexport function createSlotConfig<T extends Record<string, Type<any>>>(\n  overrides: Partial<Record<keyof T, SlotValue>>,\n  defaults: T\n): Map<keyof T, SlotRegistryEntry> {\n  const config = new Map<keyof T, SlotRegistryEntry>();\n  \n  for (const key in defaults) {\n    const override = overrides[key];\n    const defaultComponent = defaults[key];\n    config.set(key, normalizeSlotValue(override, defaultComponent));\n  }\n  \n  return config;\n}\n\n/**\n * Provides slot configuration to child components via DI.\n * \n * @example\n * ```typescript\n * @Component({\n *   providers: [\n *     provideSlots({\n *       sendButton: CustomSendButton,\n *       toolbar: CustomToolbar\n *     })\n *   ]\n * })\n * ```\n */\nexport function provideSlots(slots: Record<string, Type<any>>) {\n  const slotMap = new Map<string, SlotRegistryEntry>();\n  \n  // Only accept component types in DI (templates lack view context)\n  for (const [key, value] of Object.entries(slots)) {\n    if (isComponentType(value)) {\n      slotMap.set(key, { component: value as Type<any> });\n    }\n  }\n  \n  return {\n    provide: SLOT_CONFIG,\n    useValue: slotMap\n  };\n}\n\n/**\n * Gets slot configuration from DI.\n * Must be called within an injection context.\n * \n * @example\n * ```typescript\n * export class MyComponent {\n *   slots = getSlotConfig();\n *   \n *   ngOnInit() {\n *     const sendButton = this.slots?.get('sendButton');\n *   }\n * }\n * ```\n */\nexport function getSlotConfig(): ReadonlyMap<string, SlotRegistryEntry> | null {\n  return inject(SLOT_CONFIG, { optional: true });\n}\n\n/**\n * Creates a render function for a specific slot.\n * Useful for creating reusable slot renderers.\n * \n * @example\n * ```typescript\n * const renderSendButton = createSlotRenderer(\n *   DefaultSendButton,\n *   'sendButton'\n * );\n * \n * // Later in template\n * renderSendButton(this.viewContainer, this.sendButtonOverride);\n * ```\n */\nexport function createSlotRenderer<T>(\n  defaultComponent: Type<T>,\n  slotName?: string\n) {\n  // Get config in the injection context when the renderer is created\n  const config = slotName ? getSlotConfig() : null;\n  \n  return (\n    viewContainer: ViewContainerRef,\n    slot?: SlotValue<T>,\n    props?: Partial<T>,\n    outputs?: Record<string, (event: any) => void>\n  ) => {\n    // Check DI for overrides if slot name provided\n    if (slotName && !slot && config) {\n      const entry = config.get(slotName);\n      if (entry) {\n        if (entry.component) slot = entry.component;\n        else if (entry.template) slot = entry.template;\n      }\n    }\n    \n    return renderSlot(viewContainer, {\n      slot,\n      defaultComponent,\n      props,\n      outputs\n    });\n  };\n}"]}
231
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"slot.utils.js","sourceRoot":"","sources":["../../../../src/lib/slots/slot.utils.ts"],"names":[],"mappings":"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","sourcesContent":["import { \n  Type, \n  TemplateRef, \n  ViewContainerRef,\n  ComponentRef,\n  EmbeddedViewRef,\n  Injector,\n  inject\n} from '@angular/core';\nimport { SlotValue, RenderSlotOptions, SlotRegistryEntry, SLOT_CONFIG } from './slot.types';\n\n/**\n * Renders a slot value into a ViewContainerRef.\n * This is the core utility for slot rendering.\n * \n * @param viewContainer - The ViewContainerRef to render into\n * @param options - Options for rendering the slot\n * @returns The created component or embedded view reference\n * \n * @example\n * ```typescript\n * export class MyComponent {\n *   @ViewChild('container', { read: ViewContainerRef }) container!: ViewContainerRef;\n *   \n *   renderButton() {\n *     renderSlot(this.container, {\n *       slot: this.buttonOverride,\n *       defaultComponent: DefaultButton,\n *       props: { text: 'Click me' },\n *       outputs: { click: (event) => this.handleClick(event) }\n *     });\n *   }\n * }\n * ```\n */\nexport function renderSlot<T = any>(\n  viewContainer: ViewContainerRef,\n  options: RenderSlotOptions<T>\n): ComponentRef<T> | EmbeddedViewRef<T> | null {\n  const { slot, defaultComponent, props, injector, outputs } = options;\n  \n  viewContainer.clear();\n  \n  const effectiveSlot = slot ?? defaultComponent;\n  const effectiveInjector = injector ?? viewContainer.injector;\n  \n  if (effectiveSlot instanceof TemplateRef) {\n    // TemplateRef: render template\n    return viewContainer.createEmbeddedView(effectiveSlot, {\n      $implicit: props ?? {},\n      props: props ?? {}\n    } as any);\n  } else if (isComponentType(effectiveSlot)) {\n    // Component type - wrap in try/catch for safety\n    try {\n      return createComponent(\n        viewContainer,\n        effectiveSlot as Type<T>,\n        props,\n        effectiveInjector,\n        outputs\n      );\n    } catch (error) {\n      console.warn('Failed to create component:', effectiveSlot, error);\n      // Fall through to default component\n    }\n  }\n  \n  // Default: render default component if provided\n  return defaultComponent ? createComponent(\n    viewContainer,\n    defaultComponent,\n    props,\n    effectiveInjector,\n    outputs\n  ) : null;\n}\n\n/**\n * Creates a component and applies properties.\n */\nfunction createComponent<T>(\n  viewContainer: ViewContainerRef,\n  component: Type<T>,\n  props?: Partial<T>,\n  injector?: Injector,\n  outputs?: Record<string, (event: any) => void>\n): ComponentRef<T> {\n  const componentRef = viewContainer.createComponent(component, {\n    injector\n  });\n  \n  if (props) {\n    // Apply props using setInput, but only for declared inputs\n    const cmpDef: any = (component as any).ɵcmp;\n    const declaredInputs = new Set<string>(Object.keys(cmpDef?.inputs ?? {}));\n\n    if (declaredInputs.has('props')) {\n      componentRef.setInput('props', props as any);\n    } else {\n      for (const key in props) {\n        if (declaredInputs.has(key)) {\n          const value = (props as any)[key];\n          componentRef.setInput(key, value);\n        }\n      }\n    }\n  }\n  \n  if (outputs) {\n    // Wire up output event handlers with proper cleanup\n    const instance = componentRef.instance as any;\n    const subscriptions: any[] = [];\n    \n    for (const [eventName, handler] of Object.entries(outputs)) {\n      if (instance[eventName]?.subscribe) {\n        const subscription = instance[eventName].subscribe(handler);\n        subscriptions.push(subscription);\n      }\n    }\n    \n    // Register cleanup on component destroy\n    componentRef.onDestroy(() => {\n      subscriptions.forEach(sub => sub.unsubscribe());\n    });\n  }\n  \n  // Trigger change detection\n  componentRef.changeDetectorRef.detectChanges();\n  \n  return componentRef;\n}\n\n\n/**\n * Checks if a value is a component type.\n * Simplified check - rely on try/catch for actual validation.\n */\nexport function isComponentType(value: any): boolean {\n  // Arrow functions and regular functions without a prototype are not components\n  return typeof value === 'function' && !!value.prototype;\n}\n\n/**\n * Checks if a value is a valid slot value.\n */\nexport function isSlotValue(value: any): value is SlotValue {\n  return value instanceof TemplateRef || isComponentType(value);\n}\n\n/**\n * Normalizes a slot value to a consistent format.\n */\nexport function normalizeSlotValue<T = any>(\n  value: SlotValue<T> | undefined,\n  defaultComponent: Type<T> | undefined\n): SlotRegistryEntry<T> {\n  if (!value) {\n    return { component: defaultComponent };\n  }\n  \n  if (value instanceof TemplateRef) {\n    return { template: value };\n  }\n  \n  if (isComponentType(value)) {\n    return { component: value as Type<T> };\n  }\n  \n  return { component: defaultComponent };\n}\n\n/**\n * Creates a slot configuration map for a component.\n * \n * @example\n * ```typescript\n * const slots = createSlotConfig({\n *   sendButton: CustomSendButton,\n *   toolbar: 'custom-toolbar-class',\n *   footer: footerTemplate\n * }, {\n *   sendButton: DefaultSendButton,\n *   toolbar: DefaultToolbar,\n *   footer: DefaultFooter\n * });\n * ```\n */\nexport function createSlotConfig<T extends Record<string, Type<any>>>(\n  overrides: Partial<Record<keyof T, SlotValue>>,\n  defaults: T\n): Map<keyof T, SlotRegistryEntry> {\n  const config = new Map<keyof T, SlotRegistryEntry>();\n  \n  for (const key in defaults) {\n    const override = overrides[key];\n    const defaultComponent = defaults[key];\n    config.set(key, normalizeSlotValue(override, defaultComponent));\n  }\n  \n  return config;\n}\n\n/**\n * Provides slot configuration to child components via DI.\n * \n * @example\n * ```typescript\n * @Component({\n *   providers: [\n *     provideSlots({\n *       sendButton: CustomSendButton,\n *       toolbar: CustomToolbar\n *     })\n *   ]\n * })\n * ```\n */\nexport function provideSlots(slots: Record<string, Type<any>>) {\n  const slotMap = new Map<string, SlotRegistryEntry>();\n  \n  // Only accept component types in DI (templates lack view context)\n  for (const [key, value] of Object.entries(slots)) {\n    if (isComponentType(value)) {\n      slotMap.set(key, { component: value as Type<any> });\n    }\n  }\n  \n  return {\n    provide: SLOT_CONFIG,\n    useValue: slotMap\n  };\n}\n\n/**\n * Gets slot configuration from DI.\n * Must be called within an injection context.\n * \n * @example\n * ```typescript\n * export class MyComponent {\n *   slots = getSlotConfig();\n *   \n *   ngOnInit() {\n *     const sendButton = this.slots?.get('sendButton');\n *   }\n * }\n * ```\n */\nexport function getSlotConfig(): ReadonlyMap<string, SlotRegistryEntry> | null {\n  return inject(SLOT_CONFIG, { optional: true });\n}\n\n/**\n * Creates a render function for a specific slot.\n * Useful for creating reusable slot renderers.\n * \n * @example\n * ```typescript\n * const renderSendButton = createSlotRenderer(\n *   DefaultSendButton,\n *   'sendButton'\n * );\n * \n * // Later in template\n * renderSendButton(this.viewContainer, this.sendButtonOverride);\n * ```\n */\nexport function createSlotRenderer<T>(\n  defaultComponent: Type<T>,\n  slotName?: string\n) {\n  // Get config in the injection context when the renderer is created\n  const config = slotName ? getSlotConfig() : null;\n  \n  return (\n    viewContainer: ViewContainerRef,\n    slot?: SlotValue<T>,\n    props?: Partial<T>,\n    outputs?: Record<string, (event: any) => void>\n  ) => {\n    // Check DI for overrides if slot name provided\n    if (slotName && !slot && config) {\n      const entry = config.get(slotName);\n      if (entry) {\n        if (entry.component) slot = entry.component;\n        else if (entry.template) slot = entry.template;\n      }\n    }\n    \n    return renderSlot(viewContainer, {\n      slot,\n      defaultComponent,\n      props,\n      outputs\n    });\n  };\n}\n"]}
@@ -1,4 +1,4 @@
1
- import { DestroyRef, inject, signal, computed } from "@angular/core";
1
+ import { DestroyRef, inject, signal, computed, runInInjectionContext, } from "@angular/core";
2
2
  import { toObservable } from "@angular/core/rxjs-interop";
3
3
  import { CopilotKitService } from "../core/copilotkit.service";
4
4
  import { DEFAULT_AGENT_ID } from "@copilotkitnext/shared";
@@ -137,6 +137,32 @@ export function getAgent(service, agentId) {
137
137
  const effectiveAgentId = agentId ?? DEFAULT_AGENT_ID;
138
138
  return service.copilotkit.getAgent(effectiveAgentId);
139
139
  }
140
+ /**
141
+ * Convenience wrapper for watchAgent that handles injection context.
142
+ * Useful when you need to call watchAgent outside of a constructor or field initializer.
143
+ *
144
+ * @param injector - The Angular Injector to use for injection context
145
+ * @param config - Optional configuration with agentId
146
+ * @returns Object with agent, messages, and isRunning signals plus observables
147
+ *
148
+ * @example
149
+ * ```typescript
150
+ * export class MyComponent {
151
+ * constructor(private injector: Injector) {}
152
+ *
153
+ * switchAgent(newAgentId: string) {
154
+ * // Can call outside of constructor using watchAgentWith
155
+ * const watcher = watchAgentWith(this.injector, { agentId: newAgentId });
156
+ * this.agent = watcher.agent;
157
+ * this.messages = watcher.messages;
158
+ * this.isRunning = watcher.isRunning;
159
+ * }
160
+ * }
161
+ * ```
162
+ */
163
+ export function watchAgentWith(injector, config) {
164
+ return runInInjectionContext(injector, () => watchAgent(config));
165
+ }
140
166
  /**
141
167
  * Subscribes to an agent's events with custom callbacks.
142
168
  * Returns a cleanup function that should be called to unsubscribe.
@@ -183,22 +209,4 @@ export function subscribeToAgent(service, agentId, callbacks) {
183
209
  });
184
210
  return () => subscription.unsubscribe();
185
211
  }
186
- /**
187
- * Registers an agent watcher that automatically cleans up on component destroy.
188
- * This is an alias for watchAgent with a more explicit name.
189
- * Must be called within an injection context.
190
- *
191
- * @param config - Optional configuration with agentId
192
- * @returns Object with agent, messages, and isRunning signals plus observables
193
- *
194
- * @example
195
- * ```typescript
196
- * export class MyComponent {
197
- * agentState = registerAgentWatcher({ agentId: 'my-agent' });
198
- * }
199
- * ```
200
- */
201
- export function registerAgentWatcher(config) {
202
- return watchAgent(config);
203
- }
204
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"agent.utils.js","sourceRoot":"","sources":["../../../src/utils/agent.utils.ts"],"names":[],"mappings":"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","sourcesContent":["import { DestroyRef, inject, signal, computed } from \"@angular/core\";\nimport { toObservable } from \"@angular/core/rxjs-interop\";\nimport { CopilotKitService } from \"../core/copilotkit.service\";\nimport {\n  AgentSubscriptionCallbacks,\n  AgentWatchResult,\n} from \"../core/copilotkit.types\";\nimport { AbstractAgent } from \"@ag-ui/client\";\nimport { DEFAULT_AGENT_ID } from \"@copilotkitnext/shared\";\n\n/**\n * Watches an agent and provides reactive signals for its state.\n * Must be called within an injection context.\n * Automatically cleans up when the component/service is destroyed.\n *\n * @param config - Optional configuration with agentId\n * @returns Object with agent, messages, and isRunning signals plus observables\n *\n * @example\n * ```typescript\n * export class MyComponent {\n *   // Automatically tracks agent state\n *   agentState = watchAgent({ agentId: 'my-agent' });\n *\n *   constructor() {\n *     effect(() => {\n *       const messages = this.agentState.messages();\n *       const isRunning = this.agentState.isRunning();\n *       console.log('Messages:', messages.length, 'Running:', isRunning);\n *     });\n *   }\n * }\n * ```\n */\nexport function watchAgent(config?: { agentId?: string }): AgentWatchResult {\n  // Use inject() internally to get required services\n  const service = inject(CopilotKitService);\n  const destroyRef = inject(DestroyRef);\n  const effectiveAgentId = config?.agentId ?? DEFAULT_AGENT_ID;\n\n  // Create reactive signals with tick mechanism for reliable updates\n  const agentSignal = signal<AbstractAgent | undefined>(undefined);\n  const tick = signal<number>(0);\n  const isRunningSignal = signal<boolean>(false);\n\n  // Create computed messages signal that reacts to tick changes\n  const messages = computed(() => {\n    // Access tick to ensure recomputation\n    tick();\n    const a = agentSignal();\n    if (!a) return [];\n    // Return a shallow clone to ensure change detection\n    return a.messages.map((m) => ({ ...m }));\n  });\n\n  // Get initial agent\n  const updateAgent = () => {\n    const agent = service.copilotkit.getAgent(effectiveAgentId);\n    agentSignal.set(agent);\n    return agent;\n  };\n\n  // Initial update\n  let currentAgent = updateAgent();\n\n  // Subscribe to agent changes\n  let agentSubscription: { unsubscribe: () => void } | undefined;\n\n  const subscribeToAgent = () => {\n    // Unsubscribe from previous agent if any\n    agentSubscription?.unsubscribe();\n\n    if (currentAgent) {\n      agentSubscription = currentAgent.subscribe({\n        onMessagesChanged() {\n          // Increment tick to force recomputation of messages computed\n          tick.update((v) => v + 1);\n        },\n        onStateChanged() {\n          // Increment tick to force recomputation\n          tick.update((v) => v + 1);\n        },\n        onRunInitialized() {\n          isRunningSignal.set(true);\n        },\n        onRunFinalized() {\n          isRunningSignal.set(false);\n        },\n        onRunFailed() {\n          isRunningSignal.set(false);\n        },\n      });\n    }\n  };\n\n  // Initial subscription\n  subscribeToAgent();\n\n  // Subscribe to CopilotKit changes to detect agent updates\n  const coreUnsubscribe = service.copilotkit.subscribe({\n    onRuntimeLoaded() {\n      // Re-check agent when runtime loads\n      currentAgent = updateAgent();\n      subscribeToAgent();\n    },\n    onRuntimeLoadError() {\n      // Also re-check agent on runtime load error to ensure consistency\n      currentAgent = updateAgent();\n      subscribeToAgent();\n    },\n  });\n\n  // Register cleanup\n  const unsubscribe = () => {\n    agentSubscription?.unsubscribe();\n    coreUnsubscribe(); // subscribe returns a function directly\n  };\n\n  destroyRef.onDestroy(unsubscribe);\n\n  // Create observables from signals using toObservable\n  const agent$ = toObservable(agentSignal);\n  const isRunning$ = toObservable(isRunningSignal);\n  const messages$ = toObservable(messages);\n\n  return {\n    agent: agentSignal.asReadonly(),\n    messages: messages,\n    isRunning: isRunningSignal.asReadonly(),\n    agent$,\n    messages$,\n    isRunning$,\n    unsubscribe,\n  };\n}\n\n/**\n * Gets an agent by ID without subscribing to changes.\n *\n * @param service - The CopilotKitService instance\n * @param agentId - Optional agent ID (defaults to DEFAULT_AGENT_ID)\n * @returns The agent or undefined if not found\n *\n * @example\n * ```typescript\n * export class MyComponent {\n *   constructor(private copilotkit: CopilotKitService) {}\n *\n *   getCurrentAgent() {\n *     return getAgent(this.copilotkit, 'my-agent');\n *   }\n * }\n * ```\n */\nexport function getAgent(\n  service: CopilotKitService,\n  agentId?: string\n): AbstractAgent | undefined {\n  const effectiveAgentId = agentId ?? DEFAULT_AGENT_ID;\n  return service.copilotkit.getAgent(effectiveAgentId);\n}\n\n// Re-export the type for convenience (the actual type is in copilotkit.types)\nexport type { AgentWatchResult } from \"../core/copilotkit.types\";\n\n/**\n * Subscribes to an agent's events with custom callbacks.\n * Returns a cleanup function that should be called to unsubscribe.\n *\n * @param service - The CopilotKitService instance\n * @param agentId - Optional agent ID (defaults to DEFAULT_AGENT_ID)\n * @param callbacks - Event callbacks\n * @returns Cleanup function to unsubscribe\n *\n * @example\n * ```typescript\n * export class MyComponent implements OnInit, OnDestroy {\n *   private unsubscribe?: () => void;\n *\n *   constructor(private copilotkit: CopilotKitService) {}\n *\n *   ngOnInit() {\n *     this.unsubscribe = subscribeToAgent(this.copilotkit, 'my-agent', {\n *       onRunInitialized: () => console.log('Run started'),\n *       onRunFinalized: () => console.log('Run completed'),\n *       onRunFailed: (error) => console.error('Run failed', error),\n *     });\n *   }\n *\n *   ngOnDestroy() {\n *     this.unsubscribe?.();\n *   }\n * }\n * ```\n */\nexport function subscribeToAgent(\n  service: CopilotKitService,\n  agentId?: string,\n  callbacks?: AgentSubscriptionCallbacks\n): () => void {\n  const effectiveAgentId = agentId ?? DEFAULT_AGENT_ID;\n  const agent = service.copilotkit.getAgent(effectiveAgentId);\n\n  if (!agent) {\n    // Return no-op cleanup if agent doesn't exist\n    return () => {};\n  }\n\n  const subscription = agent.subscribe({\n    onMessagesChanged: callbacks?.onMessagesChanged,\n    onStateChanged: callbacks?.onStateChanged,\n    onRunInitialized: callbacks?.onRunInitialized,\n    onRunFinalized: callbacks?.onRunFinalized,\n    onRunFailed: callbacks?.onRunFailed,\n  });\n\n  return () => subscription.unsubscribe();\n}\n\n/**\n * Registers an agent watcher that automatically cleans up on component destroy.\n * This is an alias for watchAgent with a more explicit name.\n * Must be called within an injection context.\n *\n * @param config - Optional configuration with agentId\n * @returns Object with agent, messages, and isRunning signals plus observables\n *\n * @example\n * ```typescript\n * export class MyComponent {\n *   agentState = registerAgentWatcher({ agentId: 'my-agent' });\n * }\n * ```\n */\nexport function registerAgentWatcher(config?: {\n  agentId?: string;\n}): AgentWatchResult {\n  return watchAgent(config);\n}\n"]}
212
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"agent.utils.js","sourceRoot":"","sources":["../../../src/utils/agent.utils.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  DestroyRef,\n  inject,\n  signal,\n  computed,\n  Injector,\n  runInInjectionContext,\n} from \"@angular/core\";\nimport { toObservable } from \"@angular/core/rxjs-interop\";\nimport { CopilotKitService } from \"../core/copilotkit.service\";\nimport {\n  AgentSubscriptionCallbacks,\n  AgentWatchResult,\n} from \"../core/copilotkit.types\";\nimport { AbstractAgent } from \"@ag-ui/client\";\nimport { DEFAULT_AGENT_ID } from \"@copilotkitnext/shared\";\n\n/**\n * Watches an agent and provides reactive signals for its state.\n * Must be called within an injection context.\n * Automatically cleans up when the component/service is destroyed.\n *\n * @param config - Optional configuration with agentId\n * @returns Object with agent, messages, and isRunning signals plus observables\n *\n * @example\n * ```typescript\n * export class MyComponent {\n *   // Automatically tracks agent state\n *   agentState = watchAgent({ agentId: 'my-agent' });\n *\n *   constructor() {\n *     effect(() => {\n *       const messages = this.agentState.messages();\n *       const isRunning = this.agentState.isRunning();\n *       console.log('Messages:', messages.length, 'Running:', isRunning);\n *     });\n *   }\n * }\n * ```\n */\nexport function watchAgent(config?: { agentId?: string }): AgentWatchResult {\n  // Use inject() internally to get required services\n  const service = inject(CopilotKitService);\n  const destroyRef = inject(DestroyRef);\n  const effectiveAgentId = config?.agentId ?? DEFAULT_AGENT_ID;\n\n  // Create reactive signals with tick mechanism for reliable updates\n  const agentSignal = signal<AbstractAgent | undefined>(undefined);\n  const tick = signal<number>(0);\n  const isRunningSignal = signal<boolean>(false);\n\n  // Create computed messages signal that reacts to tick changes\n  const messages = computed(() => {\n    // Access tick to ensure recomputation\n    tick();\n    const a = agentSignal();\n    if (!a) return [];\n    // Return a shallow clone to ensure change detection\n    return a.messages.map((m) => ({ ...m }));\n  });\n\n  // Get initial agent\n  const updateAgent = () => {\n    const agent = service.copilotkit.getAgent(effectiveAgentId);\n    agentSignal.set(agent);\n    return agent;\n  };\n\n  // Initial update\n  let currentAgent = updateAgent();\n\n  // Subscribe to agent changes\n  let agentSubscription: { unsubscribe: () => void } | undefined;\n\n  const subscribeToAgent = () => {\n    // Unsubscribe from previous agent if any\n    agentSubscription?.unsubscribe();\n\n    if (currentAgent) {\n      agentSubscription = currentAgent.subscribe({\n        onMessagesChanged() {\n          // Increment tick to force recomputation of messages computed\n          tick.update((v) => v + 1);\n        },\n        onStateChanged() {\n          // Increment tick to force recomputation\n          tick.update((v) => v + 1);\n        },\n        onRunInitialized() {\n          isRunningSignal.set(true);\n        },\n        onRunFinalized() {\n          isRunningSignal.set(false);\n        },\n        onRunFailed() {\n          isRunningSignal.set(false);\n        },\n      });\n    }\n  };\n\n  // Initial subscription\n  subscribeToAgent();\n\n  // Subscribe to CopilotKit changes to detect agent updates\n  const coreUnsubscribe = service.copilotkit.subscribe({\n    onRuntimeLoaded() {\n      // Re-check agent when runtime loads\n      currentAgent = updateAgent();\n      subscribeToAgent();\n    },\n    onRuntimeLoadError() {\n      // Also re-check agent on runtime load error to ensure consistency\n      currentAgent = updateAgent();\n      subscribeToAgent();\n    },\n  });\n\n  // Register cleanup\n  const unsubscribe = () => {\n    agentSubscription?.unsubscribe();\n    coreUnsubscribe(); // subscribe returns a function directly\n  };\n\n  destroyRef.onDestroy(unsubscribe);\n\n  // Create observables from signals using toObservable\n  const agent$ = toObservable(agentSignal);\n  const isRunning$ = toObservable(isRunningSignal);\n  const messages$ = toObservable(messages);\n\n  return {\n    agent: agentSignal.asReadonly(),\n    messages: messages,\n    isRunning: isRunningSignal.asReadonly(),\n    agent$,\n    messages$,\n    isRunning$,\n    unsubscribe,\n  };\n}\n\n/**\n * Gets an agent by ID without subscribing to changes.\n *\n * @param service - The CopilotKitService instance\n * @param agentId - Optional agent ID (defaults to DEFAULT_AGENT_ID)\n * @returns The agent or undefined if not found\n *\n * @example\n * ```typescript\n * export class MyComponent {\n *   constructor(private copilotkit: CopilotKitService) {}\n *\n *   getCurrentAgent() {\n *     return getAgent(this.copilotkit, 'my-agent');\n *   }\n * }\n * ```\n */\nexport function getAgent(\n  service: CopilotKitService,\n  agentId?: string\n): AbstractAgent | undefined {\n  const effectiveAgentId = agentId ?? DEFAULT_AGENT_ID;\n  return service.copilotkit.getAgent(effectiveAgentId);\n}\n\n// Re-export the type for convenience (the actual type is in copilotkit.types)\nexport type { AgentWatchResult } from \"../core/copilotkit.types\";\n\n/**\n * Convenience wrapper for watchAgent that handles injection context.\n * Useful when you need to call watchAgent outside of a constructor or field initializer.\n *\n * @param injector - The Angular Injector to use for injection context\n * @param config - Optional configuration with agentId\n * @returns Object with agent, messages, and isRunning signals plus observables\n *\n * @example\n * ```typescript\n * export class MyComponent {\n *   constructor(private injector: Injector) {}\n *\n *   switchAgent(newAgentId: string) {\n *     // Can call outside of constructor using watchAgentWith\n *     const watcher = watchAgentWith(this.injector, { agentId: newAgentId });\n *     this.agent = watcher.agent;\n *     this.messages = watcher.messages;\n *     this.isRunning = watcher.isRunning;\n *   }\n * }\n * ```\n */\nexport function watchAgentWith(\n  injector: Injector,\n  config?: { agentId?: string }\n): AgentWatchResult {\n  return runInInjectionContext(injector, () => watchAgent(config));\n}\n\n/**\n * Subscribes to an agent's events with custom callbacks.\n * Returns a cleanup function that should be called to unsubscribe.\n *\n * @param service - The CopilotKitService instance\n * @param agentId - Optional agent ID (defaults to DEFAULT_AGENT_ID)\n * @param callbacks - Event callbacks\n * @returns Cleanup function to unsubscribe\n *\n * @example\n * ```typescript\n * export class MyComponent implements OnInit, OnDestroy {\n *   private unsubscribe?: () => void;\n *\n *   constructor(private copilotkit: CopilotKitService) {}\n *\n *   ngOnInit() {\n *     this.unsubscribe = subscribeToAgent(this.copilotkit, 'my-agent', {\n *       onRunInitialized: () => console.log('Run started'),\n *       onRunFinalized: () => console.log('Run completed'),\n *       onRunFailed: (error) => console.error('Run failed', error),\n *     });\n *   }\n *\n *   ngOnDestroy() {\n *     this.unsubscribe?.();\n *   }\n * }\n * ```\n */\nexport function subscribeToAgent(\n  service: CopilotKitService,\n  agentId?: string,\n  callbacks?: AgentSubscriptionCallbacks\n): () => void {\n  const effectiveAgentId = agentId ?? DEFAULT_AGENT_ID;\n  const agent = service.copilotkit.getAgent(effectiveAgentId);\n\n  if (!agent) {\n    // Return no-op cleanup if agent doesn't exist\n    return () => {};\n  }\n\n  const subscription = agent.subscribe({\n    onMessagesChanged: callbacks?.onMessagesChanged,\n    onStateChanged: callbacks?.onStateChanged,\n    onRunInitialized: callbacks?.onRunInitialized,\n    onRunFinalized: callbacks?.onRunFinalized,\n    onRunFailed: callbacks?.onRunFailed,\n  });\n\n  return () => subscription.unsubscribe();\n}\n"]}
@@ -1,5 +1,5 @@
1
1
  import * as i0 from '@angular/core';
2
- import { InjectionToken, signal, computed, effect, untracked, Inject, Injectable, Optional, inject, DestroyRef, ChangeDetectionStrategy, Component, ElementRef, ViewContainerRef, HostListener, Input, Directive, isDevMode, EventEmitter, Output, TemplateRef, ViewChild, ViewEncapsulation, ContentChild, ChangeDetectorRef, PLATFORM_ID, forwardRef, runInInjectionContext, SkipSelf } from '@angular/core';
2
+ import { InjectionToken, signal, computed, effect, untracked, Inject, Injectable, Optional, inject, DestroyRef, runInInjectionContext, ChangeDetectionStrategy, Component, ElementRef, ViewContainerRef, HostListener, Input, Directive, isDevMode, EventEmitter, Output, TemplateRef, ViewChild, ViewEncapsulation, ContentChild, ChangeDetectorRef, PLATFORM_ID, forwardRef, SkipSelf } from '@angular/core';
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';
@@ -259,16 +259,7 @@ class CopilotKitService {
259
259
  });
260
260
  }
261
261
  });
262
- // Warn if renderToolCalls changes
263
- effect(() => {
264
- const current = this._renderToolCalls();
265
- if (current !== this.initialRenderToolCalls &&
266
- this.initialRenderToolCalls.length > 0) {
267
- untracked(() => {
268
- console.error("renderToolCalls must be a stable object. To add/remove tools dynamically, use dynamic tool registration.");
269
- });
270
- }
271
- });
262
+ // Previously warned if renderToolCalls reference changed; removed per UX feedback
272
263
  }
273
264
  /**
274
265
  * Setup effects to sync runtime configuration with CopilotKitCore
@@ -366,9 +357,6 @@ class CopilotKitService {
366
357
  * Update render tool calls (warns if object reference changes)
367
358
  */
368
359
  setRenderToolCalls(renderToolCalls) {
369
- if (renderToolCalls !== this.initialRenderToolCalls) {
370
- console.error("renderToolCalls must be a stable object. To add/remove tools dynamically, use dynamic tool registration.");
371
- }
372
360
  this._renderToolCalls.set(renderToolCalls);
373
361
  }
374
362
  /**
@@ -1203,6 +1191,32 @@ function getAgent(service, agentId) {
1203
1191
  const effectiveAgentId = agentId ?? DEFAULT_AGENT_ID;
1204
1192
  return service.copilotkit.getAgent(effectiveAgentId);
1205
1193
  }
1194
+ /**
1195
+ * Convenience wrapper for watchAgent that handles injection context.
1196
+ * Useful when you need to call watchAgent outside of a constructor or field initializer.
1197
+ *
1198
+ * @param injector - The Angular Injector to use for injection context
1199
+ * @param config - Optional configuration with agentId
1200
+ * @returns Object with agent, messages, and isRunning signals plus observables
1201
+ *
1202
+ * @example
1203
+ * ```typescript
1204
+ * export class MyComponent {
1205
+ * constructor(private injector: Injector) {}
1206
+ *
1207
+ * switchAgent(newAgentId: string) {
1208
+ * // Can call outside of constructor using watchAgentWith
1209
+ * const watcher = watchAgentWith(this.injector, { agentId: newAgentId });
1210
+ * this.agent = watcher.agent;
1211
+ * this.messages = watcher.messages;
1212
+ * this.isRunning = watcher.isRunning;
1213
+ * }
1214
+ * }
1215
+ * ```
1216
+ */
1217
+ function watchAgentWith(injector, config) {
1218
+ return runInInjectionContext(injector, () => watchAgent(config));
1219
+ }
1206
1220
  /**
1207
1221
  * Subscribes to an agent's events with custom callbacks.
1208
1222
  * Returns a cleanup function that should be called to unsubscribe.
@@ -1249,24 +1263,6 @@ function subscribeToAgent(service, agentId, callbacks) {
1249
1263
  });
1250
1264
  return () => subscription.unsubscribe();
1251
1265
  }
1252
- /**
1253
- * Registers an agent watcher that automatically cleans up on component destroy.
1254
- * This is an alias for watchAgent with a more explicit name.
1255
- * Must be called within an injection context.
1256
- *
1257
- * @param config - Optional configuration with agentId
1258
- * @returns Object with agent, messages, and isRunning signals plus observables
1259
- *
1260
- * @example
1261
- * ```typescript
1262
- * export class MyComponent {
1263
- * agentState = registerAgentWatcher({ agentId: 'my-agent' });
1264
- * }
1265
- * ```
1266
- */
1267
- function registerAgentWatcher(config) {
1268
- return watchAgent(config);
1269
- }
1270
1266
 
1271
1267
  /**
1272
1268
  * Registers a human-in-the-loop tool that requires user interaction.
@@ -3039,20 +3035,17 @@ class CopilotKitToolRenderComponent {
3039
3035
  this.container.clear();
3040
3036
  // Create the component
3041
3037
  this.componentRef = this.container.createComponent(componentClass);
3042
- // Set inputs on the component using setInput
3043
- // Try setting a single 'props' input first
3044
- try {
3038
+ // Determine declared inputs to avoid NG0303 logs
3039
+ const cmpDef = componentClass.ɵcmp;
3040
+ const declaredInputs = new Set(Object.keys(cmpDef?.inputs ?? {}));
3041
+ if (declaredInputs.has('props')) {
3045
3042
  this.componentRef.setInput('props', props);
3046
3043
  }
3047
- catch (e) {
3048
- // If props input doesn't exist, try setting individual inputs
3044
+ else {
3049
3045
  for (const [key, value] of Object.entries(props)) {
3050
- try {
3046
+ if (declaredInputs.has(key)) {
3051
3047
  this.componentRef.setInput(key, value);
3052
3048
  }
3053
- catch (inputError) {
3054
- // Input might not exist on the component, skip it
3055
- }
3056
3049
  }
3057
3050
  }
3058
3051
  // Trigger change detection
@@ -3192,10 +3185,19 @@ function createComponent(viewContainer, component, props, injector, outputs) {
3192
3185
  injector
3193
3186
  });
3194
3187
  if (props) {
3195
- // Apply props using setInput
3196
- for (const key in props) {
3197
- const value = props[key];
3198
- componentRef.setInput(key, value);
3188
+ // Apply props using setInput, but only for declared inputs
3189
+ const cmpDef = component.ɵcmp;
3190
+ const declaredInputs = new Set(Object.keys(cmpDef?.inputs ?? {}));
3191
+ if (declaredInputs.has('props')) {
3192
+ componentRef.setInput('props', props);
3193
+ }
3194
+ else {
3195
+ for (const key in props) {
3196
+ if (declaredInputs.has(key)) {
3197
+ const value = props[key];
3198
+ componentRef.setInput(key, value);
3199
+ }
3200
+ }
3199
3201
  }
3200
3202
  }
3201
3203
  if (outputs) {
@@ -3430,11 +3432,21 @@ class CopilotSlotComponent {
3430
3432
  return;
3431
3433
  }
3432
3434
  const props = this.context;
3433
- // Update props using setInput
3435
+ // Update props using setInput, only for declared inputs
3434
3436
  if (props) {
3435
- for (const key in props) {
3436
- const value = props[key];
3437
- this.componentRef.setInput(key, value);
3437
+ const ctor = this.componentRef.instance.constructor;
3438
+ const cmpDef = ctor?.ɵcmp;
3439
+ const declaredInputs = new Set(Object.keys(cmpDef?.inputs ?? {}));
3440
+ if (declaredInputs.has('props')) {
3441
+ this.componentRef.setInput('props', props);
3442
+ }
3443
+ else {
3444
+ for (const key in props) {
3445
+ if (declaredInputs.has(key)) {
3446
+ const value = props[key];
3447
+ this.componentRef.setInput(key, value);
3448
+ }
3449
+ }
3438
3450
  }
3439
3451
  }
3440
3452
  // Trigger change detection
@@ -5940,13 +5952,19 @@ class CopilotChatToolCallsViewComponent {
5940
5952
  // Create the component
5941
5953
  const componentRef = this.container.createComponent(componentClass);
5942
5954
  this.componentRefs.set(toolCallId, componentRef);
5943
- // Set inputs on the component
5944
- for (const [key, value] of Object.entries(props)) {
5945
- try {
5946
- componentRef.setInput(key, value);
5947
- }
5948
- catch (e) {
5949
- // Input might not exist on the component, which is fine
5955
+ // Determine declared inputs to avoid Angular NG0303 console errors
5956
+ const cmpDef = componentClass.ɵcmp;
5957
+ const declaredInputs = new Set(Object.keys(cmpDef?.inputs ?? {}));
5958
+ // Prefer a single 'props' input if declared
5959
+ if (declaredInputs.has('props')) {
5960
+ componentRef.setInput('props', props);
5961
+ }
5962
+ else {
5963
+ // Otherwise, set only inputs that the component actually declares
5964
+ for (const [key, value] of Object.entries(props)) {
5965
+ if (declaredInputs.has(key)) {
5966
+ componentRef.setInput(key, value);
5967
+ }
5950
5968
  }
5951
5969
  }
5952
5970
  // Trigger change detection
@@ -6790,6 +6808,12 @@ class CopilotChatAssistantMessageComponent {
6790
6808
  toolbarContext = computed(() => ({
6791
6809
  children: null // Will be populated by the toolbar content
6792
6810
  }));
6811
+ // Return true if assistant message has non-empty text content
6812
+ hasMessageContent() {
6813
+ const raw = (this.message?.content ?? '');
6814
+ const content = typeof raw === 'string' ? raw : String(raw ?? '');
6815
+ return content.trim().length > 0;
6816
+ }
6793
6817
  toolCallsViewContext = computed(() => ({
6794
6818
  message: this.message,
6795
6819
  messages: this.messages,
@@ -6847,8 +6871,8 @@ class CopilotChatAssistantMessageComponent {
6847
6871
  </copilot-chat-tool-calls-view>
6848
6872
  }
6849
6873
 
6850
- <!-- Toolbar -->
6851
- <ng-container *ngIf="toolbarVisible">
6874
+ <!-- Toolbar: show only when there is assistant text content -->
6875
+ <ng-container *ngIf="toolbarVisible && hasMessageContent()">
6852
6876
  @if (toolbarTemplate || toolbarComponent) {
6853
6877
  <copilot-slot
6854
6878
  [slot]="toolbarTemplate || toolbarComponent"
@@ -6975,8 +6999,8 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
6975
6999
  </copilot-chat-tool-calls-view>
6976
7000
  }
6977
7001
 
6978
- <!-- Toolbar -->
6979
- <ng-container *ngIf="toolbarVisible">
7002
+ <!-- Toolbar: show only when there is assistant text content -->
7003
+ <ng-container *ngIf="toolbarVisible && hasMessageContent()">
6980
7004
  @if (toolbarTemplate || toolbarComponent) {
6981
7005
  <copilot-slot
6982
7006
  [slot]="toolbarTemplate || toolbarComponent"
@@ -9116,19 +9140,25 @@ class CopilotChatComponent {
9116
9140
  a.threadId = this.threadId || this.generatedThreadId;
9117
9141
  if (!this.hasConnectedOnce) {
9118
9142
  this.hasConnectedOnce = true;
9119
- this.connectToAgent(a);
9143
+ if ('isCopilotKitAgent' in a) {
9144
+ this.connectToAgent(a);
9145
+ }
9146
+ else {
9147
+ // Not a CopilotKit agent: ensure UI not showing loading cursor
9148
+ this.showCursor.set(false);
9149
+ this.cdr.markForCheck();
9150
+ }
9120
9151
  }
9121
9152
  }, { allowSignalWrites: true });
9122
9153
  }
9123
- // Signals from watchAgent - using direct references instead of assignment
9124
- agent = signal(undefined).asReadonly();
9125
- messages = signal([]).asReadonly();
9126
- isRunning = signal(false).asReadonly();
9154
+ // Signals from watchAgent - destructured from watcher
9155
+ agent;
9156
+ messages;
9157
+ isRunning;
9127
9158
  showCursor = signal(false);
9128
9159
  generatedThreadId = randomUUID();
9129
- agentWatcher;
9160
+ watcher;
9130
9161
  hasConnectedOnce = false;
9131
- lastAgentId;
9132
9162
  ngOnInit() {
9133
9163
  this.setupChatHandlers();
9134
9164
  }
@@ -9215,24 +9245,18 @@ class CopilotChatComponent {
9215
9245
  // Keep input state if needed
9216
9246
  });
9217
9247
  }
9218
- ngOnDestroy() {
9219
- if (this.agentWatcher?.unsubscribe) {
9220
- this.agentWatcher.unsubscribe();
9221
- }
9222
- }
9223
9248
  createWatcher(desiredAgentId) {
9224
- // Tear down previous watcher if it exists
9225
- if (this.agentWatcher?.unsubscribe) {
9226
- this.agentWatcher.unsubscribe();
9227
- this.agentWatcher = undefined;
9228
- }
9229
- // Setup watcher for desired agent - ensure injection context
9230
- this.agentWatcher = runInInjectionContext(this.injector, () => watchAgent({ agentId: desiredAgentId }));
9231
- this.agent = this.agentWatcher.agent;
9232
- this.messages = this.agentWatcher.messages;
9233
- this.isRunning = this.agentWatcher.isRunning;
9249
+ // Tear down previous watcher if it exists to prevent parallel subscriptions
9250
+ this.watcher?.unsubscribe();
9251
+ // Create new watcher using the ergonomic helper
9252
+ const w = watchAgentWith(this.injector, { agentId: desiredAgentId });
9253
+ // Destructure signals directly to class fields
9254
+ this.agent = w.agent;
9255
+ this.messages = w.messages;
9256
+ this.isRunning = w.isRunning;
9257
+ this.watcher = w;
9258
+ // Reset connection state for new agent
9234
9259
  this.hasConnectedOnce = false;
9235
- this.lastAgentId = desiredAgentId;
9236
9260
  }
9237
9261
  static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CopilotChatComponent, deps: [{ token: CopilotChatConfigurationService, optional: true }, { token: i0.ChangeDetectorRef }, { token: i0.Injector }], target: i0.ɵɵFactoryTarget.Component });
9238
9262
  static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: CopilotChatComponent, isStandalone: true, selector: "copilot-chat", inputs: { agentId: "agentId", threadId: "threadId" }, providers: [
@@ -9297,5 +9321,5 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
9297
9321
  * Generated bundle index. Do not edit.
9298
9322
  */
9299
9323
 
9300
- export { AudioRecorderError, COPILOTKIT_AGENTS, COPILOTKIT_FRONTEND_TOOLS, COPILOTKIT_HEADERS, COPILOTKIT_HUMAN_IN_THE_LOOP, COPILOTKIT_PROPERTIES, COPILOTKIT_RENDER_TOOL_CALLS, COPILOTKIT_RUNTIME_URL, COPILOT_CHAT_DEFAULT_LABELS, COPILOT_CHAT_INITIAL_CONFIG, CopilotChatAddFileButtonComponent, CopilotChatAssistantMessageComponent, CopilotChatAssistantMessageCopyButtonComponent, CopilotChatAssistantMessageReadAloudButtonComponent, CopilotChatAssistantMessageRegenerateButtonComponent, CopilotChatAssistantMessageRendererComponent, CopilotChatAssistantMessageThumbsDownButtonComponent, CopilotChatAssistantMessageThumbsUpButtonComponent, CopilotChatAssistantMessageToolbarButtonComponent, CopilotChatAssistantMessageToolbarComponent, CopilotChatAudioRecorderComponent, CopilotChatCancelTranscribeButtonComponent, CopilotChatComponent, CopilotChatConfigurationService, CopilotChatFinishTranscribeButtonComponent, CopilotChatInputComponent, CopilotChatInputDefaults, CopilotChatMessageViewComponent, CopilotChatMessageViewCursorComponent, CopilotChatSendButtonComponent, CopilotChatStartTranscribeButtonComponent, CopilotChatTextareaComponent, CopilotChatToolCallsViewComponent, CopilotChatToolbarButtonComponent, CopilotChatToolbarComponent, CopilotChatToolsMenuComponent, CopilotChatUserMessageBranchNavigationComponent, CopilotChatUserMessageComponent, CopilotChatUserMessageCopyButtonComponent, CopilotChatUserMessageEditButtonComponent, CopilotChatUserMessageRendererComponent, CopilotChatUserMessageToolbarButtonComponent, CopilotChatUserMessageToolbarComponent, CopilotChatViewComponent, CopilotChatViewDisclaimerComponent, CopilotChatViewFeatherComponent, CopilotChatViewInputContainerComponent, CopilotChatViewScrollToBottomButtonComponent, CopilotChatViewScrollViewComponent, CopilotKitAgentContextDirective, CopilotKitAgentDirective, CopilotKitChatConfigDirective, CopilotKitConfigDirective, CopilotKitFrontendToolDirective, CopilotKitHumanInTheLoopDirective, CopilotKitHumanInTheLoopRespondDirective, CopilotKitService, CopilotKitToolRenderComponent, CopilotTooltipDirective, ResizeObserverService, ScrollPositionService, StickToBottomDirective, addAgentContext, addFrontendTool, addHumanInTheLoop, createChatConfigController, createDynamicFrontendTool, createHumanInTheLoop, createReactiveContext, enhancePropsForHumanInTheLoop, getAgent, getChatInputValue, getChatLabels, injectCopilotKit, provideCopilotChatConfiguration, provideCopilotKit, registerAgentContext, registerAgentWatcher, registerChatConfig, registerFrontendTool, registerHumanInTheLoop, removeFrontendTool, setChatInputValue, setChatLabels, subscribeToAgent, watchAgent, watchChatConfig };
9324
+ export { AudioRecorderError, COPILOTKIT_AGENTS, COPILOTKIT_FRONTEND_TOOLS, COPILOTKIT_HEADERS, COPILOTKIT_HUMAN_IN_THE_LOOP, COPILOTKIT_PROPERTIES, COPILOTKIT_RENDER_TOOL_CALLS, COPILOTKIT_RUNTIME_URL, COPILOT_CHAT_DEFAULT_LABELS, COPILOT_CHAT_INITIAL_CONFIG, CopilotChatAddFileButtonComponent, CopilotChatAssistantMessageComponent, CopilotChatAssistantMessageCopyButtonComponent, CopilotChatAssistantMessageReadAloudButtonComponent, CopilotChatAssistantMessageRegenerateButtonComponent, CopilotChatAssistantMessageRendererComponent, CopilotChatAssistantMessageThumbsDownButtonComponent, CopilotChatAssistantMessageThumbsUpButtonComponent, CopilotChatAssistantMessageToolbarButtonComponent, CopilotChatAssistantMessageToolbarComponent, CopilotChatAudioRecorderComponent, CopilotChatCancelTranscribeButtonComponent, CopilotChatComponent, CopilotChatConfigurationService, CopilotChatFinishTranscribeButtonComponent, CopilotChatInputComponent, CopilotChatInputDefaults, CopilotChatMessageViewComponent, CopilotChatMessageViewCursorComponent, CopilotChatSendButtonComponent, CopilotChatStartTranscribeButtonComponent, CopilotChatTextareaComponent, CopilotChatToolCallsViewComponent, CopilotChatToolbarButtonComponent, CopilotChatToolbarComponent, CopilotChatToolsMenuComponent, CopilotChatUserMessageBranchNavigationComponent, CopilotChatUserMessageComponent, CopilotChatUserMessageCopyButtonComponent, CopilotChatUserMessageEditButtonComponent, CopilotChatUserMessageRendererComponent, CopilotChatUserMessageToolbarButtonComponent, CopilotChatUserMessageToolbarComponent, CopilotChatViewComponent, CopilotChatViewDisclaimerComponent, CopilotChatViewFeatherComponent, CopilotChatViewInputContainerComponent, CopilotChatViewScrollToBottomButtonComponent, CopilotChatViewScrollViewComponent, CopilotKitAgentContextDirective, CopilotKitAgentDirective, CopilotKitChatConfigDirective, CopilotKitConfigDirective, CopilotKitFrontendToolDirective, CopilotKitHumanInTheLoopDirective, CopilotKitHumanInTheLoopRespondDirective, CopilotKitService, CopilotKitToolRenderComponent, CopilotTooltipDirective, ResizeObserverService, ScrollPositionService, StickToBottomDirective, addAgentContext, addFrontendTool, addHumanInTheLoop, createChatConfigController, createDynamicFrontendTool, createHumanInTheLoop, createReactiveContext, enhancePropsForHumanInTheLoop, getAgent, getChatInputValue, getChatLabels, injectCopilotKit, provideCopilotChatConfiguration, provideCopilotKit, registerAgentContext, registerChatConfig, registerFrontendTool, registerHumanInTheLoop, removeFrontendTool, setChatInputValue, setChatLabels, subscribeToAgent, watchAgent, watchAgentWith, watchChatConfig };
9301
9325
  //# sourceMappingURL=copilotkitnext-angular.mjs.map