@copilotkitnext/angular 0.0.9-alpha.2 → 0.0.10
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/esm2022/index.mjs +2 -70
- package/dist/esm2022/lib/agent.mjs +73 -0
- package/dist/esm2022/lib/chat-config.mjs +35 -0
- package/dist/esm2022/lib/chat-state.mjs +18 -0
- package/dist/esm2022/lib/components/chat/copilot-chat-assistant-message-buttons.mjs +344 -0
- package/dist/esm2022/lib/components/chat/copilot-chat-assistant-message-renderer.mjs +260 -0
- package/dist/esm2022/lib/components/chat/copilot-chat-assistant-message-toolbar.mjs +22 -0
- package/dist/esm2022/{components/chat/copilot-chat-assistant-message.component.mjs → lib/components/chat/copilot-chat-assistant-message.mjs} +216 -240
- package/dist/esm2022/lib/components/chat/copilot-chat-assistant-message.types.mjs +2 -0
- package/dist/esm2022/lib/components/chat/copilot-chat-audio-recorder.mjs +196 -0
- package/dist/esm2022/lib/components/chat/copilot-chat-buttons.mjs +299 -0
- package/dist/esm2022/lib/components/chat/copilot-chat-input-defaults.mjs +39 -0
- package/dist/esm2022/lib/components/chat/copilot-chat-input.mjs +634 -0
- package/dist/esm2022/lib/components/chat/copilot-chat-input.types.mjs +10 -0
- package/dist/esm2022/lib/components/chat/copilot-chat-message-view-cursor.mjs +27 -0
- package/dist/esm2022/lib/components/chat/copilot-chat-message-view.mjs +268 -0
- package/dist/esm2022/lib/components/chat/copilot-chat-message-view.types.mjs +2 -0
- package/dist/esm2022/lib/components/chat/copilot-chat-textarea.mjs +139 -0
- package/dist/esm2022/lib/components/chat/copilot-chat-tool-calls-view.mjs +36 -0
- package/dist/esm2022/lib/components/chat/copilot-chat-toolbar.mjs +20 -0
- package/dist/esm2022/lib/components/chat/copilot-chat-tools-menu.mjs +203 -0
- package/dist/esm2022/lib/components/chat/copilot-chat-user-message-branch-navigation.mjs +118 -0
- package/dist/esm2022/lib/components/chat/copilot-chat-user-message-buttons.mjs +182 -0
- package/dist/esm2022/lib/components/chat/copilot-chat-user-message-renderer.mjs +28 -0
- package/dist/esm2022/lib/components/chat/copilot-chat-user-message-toolbar.mjs +25 -0
- package/dist/esm2022/lib/components/chat/copilot-chat-user-message.mjs +306 -0
- package/dist/esm2022/lib/components/chat/copilot-chat-user-message.types.mjs +2 -0
- package/dist/esm2022/lib/components/chat/copilot-chat-view-disclaimer.mjs +48 -0
- package/dist/esm2022/lib/components/chat/copilot-chat-view-feather.mjs +41 -0
- package/dist/esm2022/lib/components/chat/copilot-chat-view-handlers.mjs +19 -0
- package/dist/esm2022/lib/components/chat/copilot-chat-view-input-container.mjs +96 -0
- package/dist/esm2022/lib/components/chat/copilot-chat-view-scroll-to-bottom-button.mjs +89 -0
- package/dist/esm2022/lib/components/chat/copilot-chat-view-scroll-view.mjs +456 -0
- package/dist/esm2022/lib/components/chat/copilot-chat-view.mjs +404 -0
- package/dist/esm2022/lib/components/chat/copilot-chat-view.types.mjs +2 -0
- package/dist/esm2022/lib/components/chat/copilot-chat.mjs +167 -0
- package/dist/esm2022/lib/config.mjs +9 -0
- package/dist/esm2022/lib/copilotkit.mjs +124 -0
- package/dist/esm2022/lib/directives/copilotkit-agent-context.mjs +130 -0
- package/dist/esm2022/lib/directives/stick-to-bottom.mjs +170 -0
- package/dist/esm2022/lib/directives/tooltip.mjs +217 -0
- package/dist/esm2022/lib/human-in-the-loop.mjs +19 -0
- package/dist/esm2022/lib/render-tool-calls.mjs +131 -0
- package/dist/esm2022/lib/resize-observer.mjs +152 -0
- package/dist/esm2022/lib/scroll-position.mjs +124 -0
- package/dist/esm2022/lib/slots/copilot-slot.mjs +156 -0
- package/dist/esm2022/lib/slots/index.mjs +4 -0
- package/dist/esm2022/lib/slots/slot.types.mjs +3 -3
- package/dist/esm2022/lib/slots/slot.utils.mjs +19 -15
- package/dist/esm2022/lib/tools.mjs +31 -0
- package/dist/esm2022/lib/utils.mjs +3 -3
- package/dist/esm2022/public-api.mjs +47 -0
- package/dist/fesm2022/copilotkitnext-angular.mjs +5249 -8271
- package/dist/fesm2022/copilotkitnext-angular.mjs.map +1 -1
- package/dist/index.d.ts +1 -55
- package/dist/lib/agent.d.ts +53 -0
- package/dist/{core/chat-configuration/chat-configuration.types.d.ts → lib/chat-config.d.ts} +4 -8
- package/dist/lib/chat-state.d.ts +10 -0
- package/dist/lib/components/chat/copilot-chat-assistant-message-buttons.d.ts +68 -0
- package/dist/lib/components/chat/copilot-chat-assistant-message-renderer.d.ts +26 -0
- package/dist/lib/components/chat/copilot-chat-assistant-message-toolbar.d.ts +7 -0
- package/dist/lib/components/chat/copilot-chat-assistant-message.d.ts +178 -0
- package/dist/{components → lib/components}/chat/copilot-chat-assistant-message.types.d.ts +1 -1
- package/dist/{components/chat/copilot-chat-audio-recorder.component.d.ts → lib/components/chat/copilot-chat-audio-recorder.d.ts} +10 -10
- package/dist/lib/components/chat/copilot-chat-buttons.d.ts +65 -0
- package/dist/lib/components/chat/copilot-chat-input-defaults.d.ts +38 -0
- package/dist/lib/components/chat/copilot-chat-input.d.ts +133 -0
- package/dist/{components → lib/components}/chat/copilot-chat-input.types.d.ts +11 -11
- package/dist/lib/components/chat/copilot-chat-message-view-cursor.d.ts +11 -0
- package/dist/{components/chat/copilot-chat-message-view.component.d.ts → lib/components/chat/copilot-chat-message-view.d.ts} +68 -36
- package/dist/{components → lib/components}/chat/copilot-chat-message-view.types.d.ts +2 -2
- package/dist/lib/components/chat/copilot-chat-textarea.d.ts +41 -0
- package/dist/lib/components/chat/copilot-chat-tool-calls-view.d.ts +55 -0
- package/dist/lib/components/chat/copilot-chat-toolbar.d.ts +7 -0
- package/dist/lib/components/chat/copilot-chat-tools-menu.d.ts +20 -0
- package/dist/lib/components/chat/copilot-chat-user-message-branch-navigation.d.ts +20 -0
- package/dist/lib/components/chat/copilot-chat-user-message-buttons.d.ts +35 -0
- package/dist/lib/components/chat/copilot-chat-user-message-renderer.d.ts +8 -0
- package/dist/lib/components/chat/copilot-chat-user-message-toolbar.d.ts +7 -0
- package/dist/lib/components/chat/copilot-chat-user-message.d.ts +55 -0
- package/dist/{components → lib/components}/chat/copilot-chat-user-message.types.d.ts +2 -2
- package/dist/lib/components/chat/copilot-chat-view-disclaimer.d.ts +15 -0
- package/dist/{components/chat/copilot-chat-view-feather.component.d.ts → lib/components/chat/copilot-chat-view-feather.d.ts} +6 -6
- package/dist/{components/chat/copilot-chat-view-handlers.service.d.ts → lib/components/chat/copilot-chat-view-handlers.d.ts} +3 -3
- package/dist/lib/components/chat/copilot-chat-view-input-container.d.ts +23 -0
- package/dist/lib/components/chat/copilot-chat-view-scroll-to-bottom-button.d.ts +16 -0
- package/dist/lib/components/chat/copilot-chat-view-scroll-view.d.ts +114 -0
- package/dist/lib/components/chat/copilot-chat-view.d.ts +239 -0
- package/dist/{components → lib/components}/chat/copilot-chat-view.types.d.ts +2 -2
- package/dist/lib/components/chat/copilot-chat.d.ts +67 -0
- package/dist/lib/config.d.ts +16 -0
- package/dist/lib/copilotkit.d.ts +29 -0
- package/dist/{directives/copilotkit-agent-context.directive.d.ts → lib/directives/copilotkit-agent-context.d.ts} +5 -5
- package/dist/lib/directives/stick-to-bottom.d.ts +62 -0
- package/dist/lib/directives/tooltip.d.ts +33 -0
- package/dist/lib/human-in-the-loop.d.ts +13 -0
- package/dist/lib/render-tool-calls.d.ts +75 -0
- package/dist/{services/resize-observer.service.d.ts → lib/resize-observer.d.ts} +2 -2
- package/dist/{services/scroll-position.service.d.ts → lib/scroll-position.d.ts} +6 -6
- package/dist/lib/slots/copilot-slot.d.ts +34 -0
- package/dist/lib/slots/index.d.ts +3 -0
- package/dist/lib/slots/slot.types.d.ts +1 -1
- package/dist/lib/slots/slot.utils.d.ts +6 -4
- package/dist/lib/tools.d.ts +63 -0
- package/dist/lib/utils.d.ts +1 -1
- package/dist/public-api.d.ts +46 -0
- package/dist/styles.css +0 -69
- package/package.json +3 -3
- package/dist/components/chat/copilot-chat-assistant-message-buttons.component.d.ts +0 -75
- package/dist/components/chat/copilot-chat-assistant-message-renderer.component.d.ts +0 -31
- package/dist/components/chat/copilot-chat-assistant-message-toolbar.component.d.ts +0 -8
- package/dist/components/chat/copilot-chat-assistant-message.component.d.ts +0 -132
- package/dist/components/chat/copilot-chat-buttons.component.d.ts +0 -66
- package/dist/components/chat/copilot-chat-input-defaults.d.ts +0 -37
- package/dist/components/chat/copilot-chat-input.component.d.ts +0 -133
- package/dist/components/chat/copilot-chat-message-view-cursor.component.d.ts +0 -15
- package/dist/components/chat/copilot-chat-textarea.component.d.ts +0 -45
- package/dist/components/chat/copilot-chat-tool-calls-view.component.d.ts +0 -35
- package/dist/components/chat/copilot-chat-toolbar.component.d.ts +0 -8
- package/dist/components/chat/copilot-chat-tools-menu.component.d.ts +0 -20
- package/dist/components/chat/copilot-chat-user-message-branch-navigation.component.d.ts +0 -23
- package/dist/components/chat/copilot-chat-user-message-buttons.component.d.ts +0 -39
- package/dist/components/chat/copilot-chat-user-message-renderer.component.d.ts +0 -9
- package/dist/components/chat/copilot-chat-user-message-toolbar.component.d.ts +0 -8
- package/dist/components/chat/copilot-chat-user-message.component.d.ts +0 -55
- package/dist/components/chat/copilot-chat-view-disclaimer.component.d.ts +0 -15
- package/dist/components/chat/copilot-chat-view-input-container.component.d.ts +0 -23
- package/dist/components/chat/copilot-chat-view-scroll-to-bottom-button.component.d.ts +0 -17
- package/dist/components/chat/copilot-chat-view-scroll-view.component.d.ts +0 -84
- package/dist/components/chat/copilot-chat-view.component.d.ts +0 -205
- package/dist/components/chat/copilot-chat.component.d.ts +0 -36
- package/dist/components/copilotkit-tool-render.component.d.ts +0 -25
- package/dist/core/chat-configuration/chat-configuration.providers.d.ts +0 -54
- package/dist/core/chat-configuration/chat-configuration.service.d.ts +0 -75
- package/dist/core/copilotkit.providers.d.ts +0 -13
- package/dist/core/copilotkit.service.d.ts +0 -119
- package/dist/core/copilotkit.types.d.ts +0 -81
- package/dist/directives/copilotkit-agent.directive.d.ts +0 -106
- package/dist/directives/copilotkit-chat-config.directive.d.ts +0 -84
- package/dist/directives/copilotkit-config.directive.d.ts +0 -44
- package/dist/directives/copilotkit-frontend-tool.directive.d.ts +0 -25
- package/dist/directives/copilotkit-human-in-the-loop.directive.d.ts +0 -124
- package/dist/directives/stick-to-bottom.directive.d.ts +0 -62
- package/dist/esm2022/components/chat/copilot-chat-assistant-message-buttons.component.mjs +0 -384
- package/dist/esm2022/components/chat/copilot-chat-assistant-message-renderer.component.mjs +0 -286
- package/dist/esm2022/components/chat/copilot-chat-assistant-message-toolbar.component.mjs +0 -27
- package/dist/esm2022/components/chat/copilot-chat-assistant-message.types.mjs +0 -2
- package/dist/esm2022/components/chat/copilot-chat-audio-recorder.component.mjs +0 -202
- package/dist/esm2022/components/chat/copilot-chat-buttons.component.mjs +0 -321
- package/dist/esm2022/components/chat/copilot-chat-input-defaults.mjs +0 -38
- package/dist/esm2022/components/chat/copilot-chat-input.component.mjs +0 -666
- package/dist/esm2022/components/chat/copilot-chat-input.types.mjs +0 -10
- package/dist/esm2022/components/chat/copilot-chat-message-view-cursor.component.mjs +0 -45
- package/dist/esm2022/components/chat/copilot-chat-message-view.component.mjs +0 -296
- package/dist/esm2022/components/chat/copilot-chat-message-view.types.mjs +0 -2
- package/dist/esm2022/components/chat/copilot-chat-textarea.component.mjs +0 -188
- package/dist/esm2022/components/chat/copilot-chat-tool-calls-view.component.mjs +0 -222
- package/dist/esm2022/components/chat/copilot-chat-toolbar.component.mjs +0 -25
- package/dist/esm2022/components/chat/copilot-chat-tools-menu.component.mjs +0 -199
- package/dist/esm2022/components/chat/copilot-chat-user-message-branch-navigation.component.mjs +0 -137
- package/dist/esm2022/components/chat/copilot-chat-user-message-buttons.component.mjs +0 -207
- package/dist/esm2022/components/chat/copilot-chat-user-message-renderer.component.mjs +0 -35
- package/dist/esm2022/components/chat/copilot-chat-user-message-toolbar.component.mjs +0 -34
- package/dist/esm2022/components/chat/copilot-chat-user-message.component.mjs +0 -341
- package/dist/esm2022/components/chat/copilot-chat-user-message.types.mjs +0 -2
- package/dist/esm2022/components/chat/copilot-chat-view-disclaimer.component.mjs +0 -52
- package/dist/esm2022/components/chat/copilot-chat-view-feather.component.mjs +0 -55
- package/dist/esm2022/components/chat/copilot-chat-view-handlers.service.mjs +0 -19
- package/dist/esm2022/components/chat/copilot-chat-view-input-container.component.mjs +0 -110
- package/dist/esm2022/components/chat/copilot-chat-view-scroll-to-bottom-button.component.mjs +0 -93
- package/dist/esm2022/components/chat/copilot-chat-view-scroll-view.component.mjs +0 -443
- package/dist/esm2022/components/chat/copilot-chat-view.component.mjs +0 -479
- package/dist/esm2022/components/chat/copilot-chat-view.types.mjs +0 -2
- package/dist/esm2022/components/chat/copilot-chat.component.mjs +0 -220
- package/dist/esm2022/components/copilotkit-tool-render.component.mjs +0 -150
- package/dist/esm2022/core/chat-configuration/chat-configuration.providers.mjs +0 -65
- package/dist/esm2022/core/chat-configuration/chat-configuration.service.mjs +0 -145
- package/dist/esm2022/core/chat-configuration/chat-configuration.types.mjs +0 -26
- package/dist/esm2022/core/copilotkit.providers.mjs +0 -34
- package/dist/esm2022/core/copilotkit.service.mjs +0 -411
- package/dist/esm2022/core/copilotkit.types.mjs +0 -13
- package/dist/esm2022/directives/copilotkit-agent-context.directive.mjs +0 -130
- package/dist/esm2022/directives/copilotkit-agent.directive.mjs +0 -221
- package/dist/esm2022/directives/copilotkit-chat-config.directive.mjs +0 -218
- package/dist/esm2022/directives/copilotkit-config.directive.mjs +0 -94
- package/dist/esm2022/directives/copilotkit-frontend-tool.directive.mjs +0 -128
- package/dist/esm2022/directives/copilotkit-human-in-the-loop.directive.mjs +0 -265
- package/dist/esm2022/directives/stick-to-bottom.directive.mjs +0 -181
- package/dist/esm2022/lib/directives/tooltip.directive.mjs +0 -211
- package/dist/esm2022/lib/slots/copilot-slot.component.mjs +0 -154
- package/dist/esm2022/services/resize-observer.service.mjs +0 -152
- package/dist/esm2022/services/scroll-position.service.mjs +0 -124
- package/dist/esm2022/types/frontend-tool.mjs +0 -2
- package/dist/esm2022/types/human-in-the-loop.mjs +0 -2
- package/dist/esm2022/utils/agent-context.utils.mjs +0 -114
- package/dist/esm2022/utils/agent.utils.mjs +0 -212
- package/dist/esm2022/utils/chat-config.utils.mjs +0 -186
- package/dist/esm2022/utils/copilotkit.utils.mjs +0 -20
- package/dist/esm2022/utils/frontend-tool.utils.mjs +0 -224
- package/dist/esm2022/utils/human-in-the-loop.utils.mjs +0 -293
- package/dist/lib/directives/tooltip.directive.d.ts +0 -33
- package/dist/lib/slots/copilot-slot.component.d.ts +0 -34
- package/dist/types/frontend-tool.d.ts +0 -37
- package/dist/types/human-in-the-loop.d.ts +0 -44
- package/dist/utils/agent-context.utils.d.ts +0 -75
- package/dist/utils/agent.utils.d.ts +0 -108
- package/dist/utils/chat-config.utils.d.ts +0 -166
- package/dist/utils/copilotkit.utils.d.ts +0 -16
- package/dist/utils/frontend-tool.utils.d.ts +0 -119
- package/dist/utils/human-in-the-loop.utils.d.ts +0 -92
|
@@ -0,0 +1,170 @@
|
|
|
1
|
+
import { Directive, ElementRef, inject, input, output, } from "@angular/core";
|
|
2
|
+
import { ScrollPosition } from "../scroll-position";
|
|
3
|
+
import { ResizeObserverService } from "../resize-observer";
|
|
4
|
+
import { Subject } from "rxjs";
|
|
5
|
+
import { takeUntil, debounceTime, filter, distinctUntilChanged, } from "rxjs/operators";
|
|
6
|
+
import * as i0 from "@angular/core";
|
|
7
|
+
/**
|
|
8
|
+
* Directive for implementing stick-to-bottom scroll behavior
|
|
9
|
+
* Similar to the React use-stick-to-bottom library
|
|
10
|
+
*
|
|
11
|
+
* @example
|
|
12
|
+
* ```html
|
|
13
|
+
* <div copilotStickToBottom
|
|
14
|
+
* [enabled]="true"
|
|
15
|
+
* [threshold]="10"
|
|
16
|
+
* [initialBehavior]="'smooth'"
|
|
17
|
+
* [resizeBehavior]="'smooth'"
|
|
18
|
+
* (isAtBottomChange)="onBottomChange($event)">
|
|
19
|
+
* <!-- Content -->
|
|
20
|
+
* </div>
|
|
21
|
+
* ```
|
|
22
|
+
*/
|
|
23
|
+
export class StickToBottom {
|
|
24
|
+
enabled = input(true);
|
|
25
|
+
threshold = input(10);
|
|
26
|
+
initialBehavior = input("smooth");
|
|
27
|
+
resizeBehavior = input("smooth");
|
|
28
|
+
debounceMs = input(100);
|
|
29
|
+
isAtBottomChange = output();
|
|
30
|
+
scrollToBottomRequested = output();
|
|
31
|
+
elementRef = inject(ElementRef);
|
|
32
|
+
scrollService = inject(ScrollPosition);
|
|
33
|
+
resizeService = inject(ResizeObserverService);
|
|
34
|
+
destroy$ = new Subject();
|
|
35
|
+
contentElement;
|
|
36
|
+
wasAtBottom = true;
|
|
37
|
+
hasInitialized = false;
|
|
38
|
+
userHasScrolled = false;
|
|
39
|
+
ngOnInit() {
|
|
40
|
+
// Setup will happen in ngAfterViewInit
|
|
41
|
+
}
|
|
42
|
+
ngAfterViewInit() {
|
|
43
|
+
const element = this.elementRef.nativeElement;
|
|
44
|
+
// Find or create content wrapper
|
|
45
|
+
this.contentElement = element.querySelector("[data-stick-to-bottom-content]");
|
|
46
|
+
if (!this.contentElement) {
|
|
47
|
+
this.contentElement = element;
|
|
48
|
+
}
|
|
49
|
+
this.setupScrollMonitoring();
|
|
50
|
+
this.setupResizeMonitoring();
|
|
51
|
+
this.setupContentMutationObserver();
|
|
52
|
+
// Initial scroll to bottom if enabled
|
|
53
|
+
setTimeout(() => {
|
|
54
|
+
this.hasInitialized = true;
|
|
55
|
+
if (this.enabled()) {
|
|
56
|
+
this.scrollToBottom(this.initialBehavior());
|
|
57
|
+
}
|
|
58
|
+
}, 0);
|
|
59
|
+
}
|
|
60
|
+
setupScrollMonitoring() {
|
|
61
|
+
if (!this.enabled())
|
|
62
|
+
return;
|
|
63
|
+
const element = this.elementRef.nativeElement;
|
|
64
|
+
// Monitor scroll position
|
|
65
|
+
this.scrollService
|
|
66
|
+
.monitorScrollPosition(element, this.threshold())
|
|
67
|
+
.pipe(takeUntil(this.destroy$), debounceTime(this.debounceMs()), distinctUntilChanged((a, b) => a.isAtBottom === b.isAtBottom))
|
|
68
|
+
.subscribe((state) => {
|
|
69
|
+
const wasAtBottom = this.wasAtBottom;
|
|
70
|
+
this.wasAtBottom = state.isAtBottom;
|
|
71
|
+
// Detect user scroll
|
|
72
|
+
if (!state.isAtBottom && wasAtBottom && this.hasInitialized) {
|
|
73
|
+
this.userHasScrolled = true;
|
|
74
|
+
}
|
|
75
|
+
else if (state.isAtBottom) {
|
|
76
|
+
this.userHasScrolled = false;
|
|
77
|
+
}
|
|
78
|
+
// Emit change
|
|
79
|
+
this.isAtBottomChange.emit(state.isAtBottom);
|
|
80
|
+
});
|
|
81
|
+
}
|
|
82
|
+
setupResizeMonitoring() {
|
|
83
|
+
if (!this.enabled() || !this.contentElement)
|
|
84
|
+
return;
|
|
85
|
+
// Monitor content resize
|
|
86
|
+
this.resizeService
|
|
87
|
+
.observeElement(this.contentElement, 0, 250)
|
|
88
|
+
.pipe(takeUntil(this.destroy$), filter(() => this.enabled() && !this.userHasScrolled))
|
|
89
|
+
.subscribe((state) => {
|
|
90
|
+
// Auto-scroll on resize if we were at bottom
|
|
91
|
+
if (this.wasAtBottom && !state.isResizing) {
|
|
92
|
+
this.scrollToBottom(this.resizeBehavior());
|
|
93
|
+
}
|
|
94
|
+
});
|
|
95
|
+
// Monitor container resize
|
|
96
|
+
const element = this.elementRef.nativeElement;
|
|
97
|
+
this.resizeService
|
|
98
|
+
.observeElement(element, 0, 250)
|
|
99
|
+
.pipe(takeUntil(this.destroy$), filter(() => this.enabled() && !this.userHasScrolled && this.wasAtBottom))
|
|
100
|
+
.subscribe(() => {
|
|
101
|
+
// Adjust scroll on container resize
|
|
102
|
+
this.scrollToBottom(this.resizeBehavior());
|
|
103
|
+
});
|
|
104
|
+
}
|
|
105
|
+
setupContentMutationObserver() {
|
|
106
|
+
if (!this.enabled() || !this.contentElement)
|
|
107
|
+
return;
|
|
108
|
+
const mutationObserver = new MutationObserver(() => {
|
|
109
|
+
if (this.enabled() && this.wasAtBottom && !this.userHasScrolled) {
|
|
110
|
+
// Content changed, scroll to bottom if we were there
|
|
111
|
+
requestAnimationFrame(() => {
|
|
112
|
+
this.scrollToBottom(this.resizeBehavior());
|
|
113
|
+
});
|
|
114
|
+
}
|
|
115
|
+
});
|
|
116
|
+
mutationObserver.observe(this.contentElement, {
|
|
117
|
+
childList: true,
|
|
118
|
+
subtree: true,
|
|
119
|
+
characterData: true,
|
|
120
|
+
});
|
|
121
|
+
// Cleanup on destroy
|
|
122
|
+
this.destroy$.subscribe(() => {
|
|
123
|
+
mutationObserver.disconnect();
|
|
124
|
+
});
|
|
125
|
+
}
|
|
126
|
+
/**
|
|
127
|
+
* Public method to scroll to bottom
|
|
128
|
+
* Can be called from parent component
|
|
129
|
+
*/
|
|
130
|
+
scrollToBottom(behavior = "smooth") {
|
|
131
|
+
const element = this.elementRef.nativeElement;
|
|
132
|
+
const smooth = behavior === "smooth";
|
|
133
|
+
this.scrollService.scrollToBottom(element, smooth);
|
|
134
|
+
this.userHasScrolled = false;
|
|
135
|
+
this.scrollToBottomRequested.emit();
|
|
136
|
+
}
|
|
137
|
+
/**
|
|
138
|
+
* Check if currently at bottom
|
|
139
|
+
*/
|
|
140
|
+
isAtBottom() {
|
|
141
|
+
return this.scrollService.isAtBottom(this.elementRef.nativeElement, this.threshold());
|
|
142
|
+
}
|
|
143
|
+
/**
|
|
144
|
+
* Get current scroll state
|
|
145
|
+
*/
|
|
146
|
+
getScrollState() {
|
|
147
|
+
const element = this.elementRef.nativeElement;
|
|
148
|
+
return {
|
|
149
|
+
scrollTop: element.scrollTop,
|
|
150
|
+
scrollHeight: element.scrollHeight,
|
|
151
|
+
clientHeight: element.clientHeight,
|
|
152
|
+
isAtBottom: this.isAtBottom(),
|
|
153
|
+
};
|
|
154
|
+
}
|
|
155
|
+
ngOnDestroy() {
|
|
156
|
+
this.destroy$.next();
|
|
157
|
+
this.destroy$.complete();
|
|
158
|
+
}
|
|
159
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: StickToBottom, deps: [], target: i0.ɵɵFactoryTarget.Directive });
|
|
160
|
+
static ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "17.1.0", version: "18.2.13", type: StickToBottom, isStandalone: true, selector: "[copilotStickToBottom]", inputs: { enabled: { classPropertyName: "enabled", publicName: "enabled", isSignal: true, isRequired: false, transformFunction: null }, threshold: { classPropertyName: "threshold", publicName: "threshold", isSignal: true, isRequired: false, transformFunction: null }, initialBehavior: { classPropertyName: "initialBehavior", publicName: "initialBehavior", isSignal: true, isRequired: false, transformFunction: null }, resizeBehavior: { classPropertyName: "resizeBehavior", publicName: "resizeBehavior", isSignal: true, isRequired: false, transformFunction: null }, debounceMs: { classPropertyName: "debounceMs", publicName: "debounceMs", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { isAtBottomChange: "isAtBottomChange", scrollToBottomRequested: "scrollToBottomRequested" }, providers: [ScrollPosition, ResizeObserverService], ngImport: i0 });
|
|
161
|
+
}
|
|
162
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: StickToBottom, decorators: [{
|
|
163
|
+
type: Directive,
|
|
164
|
+
args: [{
|
|
165
|
+
standalone: true,
|
|
166
|
+
selector: "[copilotStickToBottom]",
|
|
167
|
+
providers: [ScrollPosition, ResizeObserverService],
|
|
168
|
+
}]
|
|
169
|
+
}] });
|
|
170
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"stick-to-bottom.js","sourceRoot":"","sources":["../../../../src/lib/directives/stick-to-bottom.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Directive,\n  ElementRef,\n  OnInit,\n  OnDestroy,\n  AfterViewInit,\n  inject,\n  input,\n  output,\n} from \"@angular/core\";\nimport { ScrollPosition } from \"../scroll-position\";\nimport { ResizeObserverService } from \"../resize-observer\";\nimport { Subject } from \"rxjs\";\nimport {\n  takeUntil,\n  debounceTime,\n  filter,\n  distinctUntilChanged,\n} from \"rxjs/operators\";\n\nexport type ScrollBehavior = \"smooth\" | \"instant\" | \"auto\";\n\n/**\n * Directive for implementing stick-to-bottom scroll behavior\n * Similar to the React use-stick-to-bottom library\n *\n * @example\n * ```html\n * <div copilotStickToBottom\n *      [enabled]=\"true\"\n *      [threshold]=\"10\"\n *      [initialBehavior]=\"'smooth'\"\n *      [resizeBehavior]=\"'smooth'\"\n *      (isAtBottomChange)=\"onBottomChange($event)\">\n *   <!-- Content -->\n * </div>\n * ```\n */\n@Directive({\n  standalone: true,\n  selector: \"[copilotStickToBottom]\",\n  providers: [ScrollPosition, ResizeObserverService],\n})\nexport class StickToBottom implements OnInit, AfterViewInit, OnDestroy {\n  enabled = input<boolean>(true);\n  threshold = input<number>(10);\n  initialBehavior = input<ScrollBehavior>(\"smooth\");\n  resizeBehavior = input<ScrollBehavior>(\"smooth\");\n  debounceMs = input<number>(100);\n\n  isAtBottomChange = output<boolean>();\n  scrollToBottomRequested = output<void>();\n\n  private elementRef = inject(ElementRef);\n  private scrollService = inject(ScrollPosition);\n  private resizeService = inject(ResizeObserverService);\n\n  private destroy$ = new Subject<void>();\n  private contentElement?: HTMLElement;\n  private wasAtBottom = true;\n  private hasInitialized = false;\n  private userHasScrolled = false;\n\n  ngOnInit(): void {\n    // Setup will happen in ngAfterViewInit\n  }\n\n  ngAfterViewInit(): void {\n    const element = this.elementRef.nativeElement as HTMLElement;\n\n    // Find or create content wrapper\n    this.contentElement = element.querySelector(\n      \"[data-stick-to-bottom-content]\"\n    ) as HTMLElement;\n    if (!this.contentElement) {\n      this.contentElement = element;\n    }\n\n    this.setupScrollMonitoring();\n    this.setupResizeMonitoring();\n    this.setupContentMutationObserver();\n\n    // Initial scroll to bottom if enabled\n    setTimeout(() => {\n      this.hasInitialized = true;\n      if (this.enabled()) {\n        this.scrollToBottom(this.initialBehavior());\n      }\n    }, 0);\n  }\n\n  private setupScrollMonitoring(): void {\n    if (!this.enabled()) return;\n\n    const element = this.elementRef.nativeElement;\n\n    // Monitor scroll position\n    this.scrollService\n      .monitorScrollPosition(element, this.threshold())\n      .pipe(\n        takeUntil(this.destroy$),\n        debounceTime(this.debounceMs()),\n        distinctUntilChanged((a, b) => a.isAtBottom === b.isAtBottom)\n      )\n      .subscribe((state) => {\n        const wasAtBottom = this.wasAtBottom;\n        this.wasAtBottom = state.isAtBottom;\n\n        // Detect user scroll\n        if (!state.isAtBottom && wasAtBottom && this.hasInitialized) {\n          this.userHasScrolled = true;\n        } else if (state.isAtBottom) {\n          this.userHasScrolled = false;\n        }\n\n        // Emit change\n        this.isAtBottomChange.emit(state.isAtBottom);\n      });\n  }\n\n  private setupResizeMonitoring(): void {\n    if (!this.enabled() || !this.contentElement) return;\n\n    // Monitor content resize\n    this.resizeService\n      .observeElement(this.contentElement, 0, 250)\n      .pipe(\n        takeUntil(this.destroy$),\n        filter(() => this.enabled() && !this.userHasScrolled)\n      )\n      .subscribe((state) => {\n        // Auto-scroll on resize if we were at bottom\n        if (this.wasAtBottom && !state.isResizing) {\n          this.scrollToBottom(this.resizeBehavior());\n        }\n      });\n\n    // Monitor container resize\n    const element = this.elementRef.nativeElement;\n    this.resizeService\n      .observeElement(element, 0, 250)\n      .pipe(\n        takeUntil(this.destroy$),\n        filter(\n          () => this.enabled() && !this.userHasScrolled && this.wasAtBottom\n        )\n      )\n      .subscribe(() => {\n        // Adjust scroll on container resize\n        this.scrollToBottom(this.resizeBehavior());\n      });\n  }\n\n  private setupContentMutationObserver(): void {\n    if (!this.enabled() || !this.contentElement) return;\n\n    const mutationObserver = new MutationObserver(() => {\n      if (this.enabled() && this.wasAtBottom && !this.userHasScrolled) {\n        // Content changed, scroll to bottom if we were there\n        requestAnimationFrame(() => {\n          this.scrollToBottom(this.resizeBehavior());\n        });\n      }\n    });\n\n    mutationObserver.observe(this.contentElement, {\n      childList: true,\n      subtree: true,\n      characterData: true,\n    });\n\n    // Cleanup on destroy\n    this.destroy$.subscribe(() => {\n      mutationObserver.disconnect();\n    });\n  }\n\n  /**\n   * Public method to scroll to bottom\n   * Can be called from parent component\n   */\n  public scrollToBottom(behavior: ScrollBehavior = \"smooth\"): void {\n    const element = this.elementRef.nativeElement;\n    const smooth = behavior === \"smooth\";\n\n    this.scrollService.scrollToBottom(element, smooth);\n    this.userHasScrolled = false;\n    this.scrollToBottomRequested.emit();\n  }\n\n  /**\n   * Check if currently at bottom\n   */\n  public isAtBottom(): boolean {\n    return this.scrollService.isAtBottom(\n      this.elementRef.nativeElement,\n      this.threshold()\n    );\n  }\n\n  /**\n   * Get current scroll state\n   */\n  public getScrollState() {\n    const element = this.elementRef.nativeElement;\n    return {\n      scrollTop: element.scrollTop,\n      scrollHeight: element.scrollHeight,\n      clientHeight: element.clientHeight,\n      isAtBottom: this.isAtBottom(),\n    };\n  }\n\n  ngOnDestroy(): void {\n    this.destroy$.next();\n    this.destroy$.complete();\n  }\n}\n"]}
|
|
@@ -0,0 +1,217 @@
|
|
|
1
|
+
import { Directive, Input, ElementRef, HostListener, inject, ViewContainerRef, } from "@angular/core";
|
|
2
|
+
import { Overlay, OverlayPositionBuilder, } from "@angular/cdk/overlay";
|
|
3
|
+
import { ComponentPortal } from "@angular/cdk/portal";
|
|
4
|
+
import { Component, ChangeDetectionStrategy, } from "@angular/core";
|
|
5
|
+
import * as i0 from "@angular/core";
|
|
6
|
+
export class TooltipContent {
|
|
7
|
+
cdr;
|
|
8
|
+
text = "";
|
|
9
|
+
_position = "below";
|
|
10
|
+
get position() {
|
|
11
|
+
return this._position;
|
|
12
|
+
}
|
|
13
|
+
set position(value) {
|
|
14
|
+
this._position = value;
|
|
15
|
+
this.cdr.markForCheck();
|
|
16
|
+
}
|
|
17
|
+
constructor(cdr) {
|
|
18
|
+
this.cdr = cdr;
|
|
19
|
+
}
|
|
20
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TooltipContent, deps: [{ token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
|
|
21
|
+
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: TooltipContent, isStandalone: true, selector: "copilot-tooltip-content", ngImport: i0, template: `
|
|
22
|
+
<div class="copilot-tooltip-wrapper" [attr.data-position]="position">
|
|
23
|
+
<div class="copilot-tooltip">
|
|
24
|
+
{{ text }}
|
|
25
|
+
</div>
|
|
26
|
+
<div class="copilot-tooltip-arrow"></div>
|
|
27
|
+
</div>
|
|
28
|
+
`, isInline: true, styles: [".copilot-tooltip-wrapper{position:relative;display:inline-block;animation:fadeIn .15s ease-in-out}.copilot-tooltip{background-color:#1a1a1a;color:#fff;padding:6px 10px;border-radius:6px;font-size:12px;font-weight:500;white-space:nowrap;max-width:200px;box-shadow:0 2px 8px #00000026}.copilot-tooltip-arrow{position:absolute;width:0;height:0;border-style:solid}.copilot-tooltip-wrapper[data-position=below] .copilot-tooltip-arrow{top:-4px;left:50%;transform:translate(-50%);border-width:0 4px 4px 4px;border-color:transparent transparent #1a1a1a transparent}.copilot-tooltip-wrapper[data-position=above] .copilot-tooltip-arrow{bottom:-4px;left:50%;transform:translate(-50%);border-width:4px 4px 0 4px;border-color:#1a1a1a transparent transparent transparent}.copilot-tooltip-wrapper[data-position=left] .copilot-tooltip-arrow{right:-4px;top:50%;transform:translateY(-50%);border-width:4px 0 4px 4px;border-color:transparent transparent transparent #1a1a1a}.copilot-tooltip-wrapper[data-position=right] .copilot-tooltip-arrow{left:-4px;top:50%;transform:translateY(-50%);border-width:4px 4px 4px 0;border-color:transparent #1a1a1a transparent transparent}@keyframes fadeIn{0%{opacity:0;transform:translateY(2px)}to{opacity:1;transform:translateY(0)}}\n"], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
29
|
+
}
|
|
30
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TooltipContent, decorators: [{
|
|
31
|
+
type: Component,
|
|
32
|
+
args: [{ selector: "copilot-tooltip-content", standalone: true, template: `
|
|
33
|
+
<div class="copilot-tooltip-wrapper" [attr.data-position]="position">
|
|
34
|
+
<div class="copilot-tooltip">
|
|
35
|
+
{{ text }}
|
|
36
|
+
</div>
|
|
37
|
+
<div class="copilot-tooltip-arrow"></div>
|
|
38
|
+
</div>
|
|
39
|
+
`, changeDetection: ChangeDetectionStrategy.OnPush, styles: [".copilot-tooltip-wrapper{position:relative;display:inline-block;animation:fadeIn .15s ease-in-out}.copilot-tooltip{background-color:#1a1a1a;color:#fff;padding:6px 10px;border-radius:6px;font-size:12px;font-weight:500;white-space:nowrap;max-width:200px;box-shadow:0 2px 8px #00000026}.copilot-tooltip-arrow{position:absolute;width:0;height:0;border-style:solid}.copilot-tooltip-wrapper[data-position=below] .copilot-tooltip-arrow{top:-4px;left:50%;transform:translate(-50%);border-width:0 4px 4px 4px;border-color:transparent transparent #1a1a1a transparent}.copilot-tooltip-wrapper[data-position=above] .copilot-tooltip-arrow{bottom:-4px;left:50%;transform:translate(-50%);border-width:4px 4px 0 4px;border-color:#1a1a1a transparent transparent transparent}.copilot-tooltip-wrapper[data-position=left] .copilot-tooltip-arrow{right:-4px;top:50%;transform:translateY(-50%);border-width:4px 0 4px 4px;border-color:transparent transparent transparent #1a1a1a}.copilot-tooltip-wrapper[data-position=right] .copilot-tooltip-arrow{left:-4px;top:50%;transform:translateY(-50%);border-width:4px 4px 4px 0;border-color:transparent #1a1a1a transparent transparent}@keyframes fadeIn{0%{opacity:0;transform:translateY(2px)}to{opacity:1;transform:translateY(0)}}\n"] }]
|
|
40
|
+
}], ctorParameters: () => [{ type: i0.ChangeDetectorRef }] });
|
|
41
|
+
export class CopilotTooltip {
|
|
42
|
+
tooltipText = "";
|
|
43
|
+
tooltipPosition = "below";
|
|
44
|
+
tooltipDelay = 500; // milliseconds
|
|
45
|
+
overlay = inject(Overlay);
|
|
46
|
+
overlayPositionBuilder = inject(OverlayPositionBuilder);
|
|
47
|
+
elementRef = inject(ElementRef);
|
|
48
|
+
viewContainerRef = inject(ViewContainerRef);
|
|
49
|
+
overlayRef;
|
|
50
|
+
tooltipTimeout;
|
|
51
|
+
originalTitle;
|
|
52
|
+
onMouseEnter() {
|
|
53
|
+
if (!this.tooltipText)
|
|
54
|
+
return;
|
|
55
|
+
// Store and remove native title to prevent OS tooltip
|
|
56
|
+
const element = this.elementRef.nativeElement;
|
|
57
|
+
if (element.hasAttribute("title")) {
|
|
58
|
+
this.originalTitle = element.getAttribute("title");
|
|
59
|
+
element.removeAttribute("title");
|
|
60
|
+
}
|
|
61
|
+
// Clear any existing timeout
|
|
62
|
+
if (this.tooltipTimeout) {
|
|
63
|
+
clearTimeout(this.tooltipTimeout);
|
|
64
|
+
}
|
|
65
|
+
// Set timeout to show tooltip after delay
|
|
66
|
+
this.tooltipTimeout = window.setTimeout(() => {
|
|
67
|
+
this.show();
|
|
68
|
+
}, this.tooltipDelay);
|
|
69
|
+
}
|
|
70
|
+
onMouseLeave() {
|
|
71
|
+
// Clear timeout if mouse leaves before tooltip shows
|
|
72
|
+
if (this.tooltipTimeout) {
|
|
73
|
+
clearTimeout(this.tooltipTimeout);
|
|
74
|
+
this.tooltipTimeout = undefined;
|
|
75
|
+
}
|
|
76
|
+
// Restore original title if it existed
|
|
77
|
+
if (this.originalTitle !== undefined) {
|
|
78
|
+
this.elementRef.nativeElement.setAttribute("title", this.originalTitle);
|
|
79
|
+
this.originalTitle = undefined;
|
|
80
|
+
}
|
|
81
|
+
// Hide tooltip if it's showing
|
|
82
|
+
this.hide();
|
|
83
|
+
}
|
|
84
|
+
show() {
|
|
85
|
+
if (this.overlayRef) {
|
|
86
|
+
return;
|
|
87
|
+
}
|
|
88
|
+
// Create overlay
|
|
89
|
+
const positionStrategy = this.overlayPositionBuilder
|
|
90
|
+
.flexibleConnectedTo(this.elementRef)
|
|
91
|
+
.withPositions(this.getPositions())
|
|
92
|
+
.withPush(false);
|
|
93
|
+
this.overlayRef = this.overlay.create({
|
|
94
|
+
positionStrategy,
|
|
95
|
+
scrollStrategy: this.overlay.scrollStrategies.close(),
|
|
96
|
+
hasBackdrop: false,
|
|
97
|
+
});
|
|
98
|
+
// Create component portal and attach
|
|
99
|
+
const portal = new ComponentPortal(TooltipContent, this.viewContainerRef);
|
|
100
|
+
const componentRef = this.overlayRef.attach(portal);
|
|
101
|
+
componentRef.instance.text = this.tooltipText;
|
|
102
|
+
// Detect actual position after overlay is positioned
|
|
103
|
+
setTimeout(() => {
|
|
104
|
+
if (this.overlayRef && this.elementRef.nativeElement) {
|
|
105
|
+
const tooltipRect = this.overlayRef.overlayElement.getBoundingClientRect();
|
|
106
|
+
const elementRect = this.elementRef.nativeElement.getBoundingClientRect();
|
|
107
|
+
let actualPosition = "below";
|
|
108
|
+
// Determine actual position based on relative positions
|
|
109
|
+
if (tooltipRect.bottom <= elementRect.top) {
|
|
110
|
+
actualPosition = "above";
|
|
111
|
+
}
|
|
112
|
+
else if (tooltipRect.top >= elementRect.bottom) {
|
|
113
|
+
actualPosition = "below";
|
|
114
|
+
}
|
|
115
|
+
else if (tooltipRect.right <= elementRect.left) {
|
|
116
|
+
actualPosition = "left";
|
|
117
|
+
}
|
|
118
|
+
else if (tooltipRect.left >= elementRect.right) {
|
|
119
|
+
actualPosition = "right";
|
|
120
|
+
}
|
|
121
|
+
componentRef.instance.position = actualPosition;
|
|
122
|
+
}
|
|
123
|
+
}, 0);
|
|
124
|
+
}
|
|
125
|
+
hide() {
|
|
126
|
+
if (this.overlayRef) {
|
|
127
|
+
this.overlayRef.dispose();
|
|
128
|
+
this.overlayRef = undefined;
|
|
129
|
+
}
|
|
130
|
+
}
|
|
131
|
+
getPositions() {
|
|
132
|
+
const positions = {
|
|
133
|
+
above: [
|
|
134
|
+
{
|
|
135
|
+
originX: "center",
|
|
136
|
+
originY: "top",
|
|
137
|
+
overlayX: "center",
|
|
138
|
+
overlayY: "bottom",
|
|
139
|
+
offsetY: -12,
|
|
140
|
+
},
|
|
141
|
+
],
|
|
142
|
+
below: [
|
|
143
|
+
{
|
|
144
|
+
originX: "center",
|
|
145
|
+
originY: "bottom",
|
|
146
|
+
overlayX: "center",
|
|
147
|
+
overlayY: "top",
|
|
148
|
+
offsetY: 12,
|
|
149
|
+
},
|
|
150
|
+
],
|
|
151
|
+
left: [
|
|
152
|
+
{
|
|
153
|
+
originX: "start",
|
|
154
|
+
originY: "center",
|
|
155
|
+
overlayX: "end",
|
|
156
|
+
overlayY: "center",
|
|
157
|
+
offsetX: -12,
|
|
158
|
+
},
|
|
159
|
+
],
|
|
160
|
+
right: [
|
|
161
|
+
{
|
|
162
|
+
originX: "end",
|
|
163
|
+
originY: "center",
|
|
164
|
+
overlayX: "start",
|
|
165
|
+
overlayY: "center",
|
|
166
|
+
offsetX: 12,
|
|
167
|
+
},
|
|
168
|
+
],
|
|
169
|
+
};
|
|
170
|
+
// Prefer below position, but use above as fallback
|
|
171
|
+
const primary = positions[this.tooltipPosition] || positions.below;
|
|
172
|
+
// For below position, add above as first fallback
|
|
173
|
+
const fallbacks = this.tooltipPosition === "below"
|
|
174
|
+
? [
|
|
175
|
+
...(positions.above || []),
|
|
176
|
+
...(positions.left || []),
|
|
177
|
+
...(positions.right || []),
|
|
178
|
+
]
|
|
179
|
+
: Object.values(positions)
|
|
180
|
+
.filter((p) => p !== primary)
|
|
181
|
+
.flat();
|
|
182
|
+
return [...(primary || []), ...fallbacks];
|
|
183
|
+
}
|
|
184
|
+
ngOnDestroy() {
|
|
185
|
+
if (this.tooltipTimeout) {
|
|
186
|
+
clearTimeout(this.tooltipTimeout);
|
|
187
|
+
}
|
|
188
|
+
// Restore original title if it existed
|
|
189
|
+
if (this.originalTitle !== undefined) {
|
|
190
|
+
this.elementRef.nativeElement.setAttribute("title", this.originalTitle);
|
|
191
|
+
}
|
|
192
|
+
this.hide();
|
|
193
|
+
}
|
|
194
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CopilotTooltip, deps: [], target: i0.ɵɵFactoryTarget.Directive });
|
|
195
|
+
static ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "18.2.13", type: CopilotTooltip, isStandalone: true, selector: "[copilotTooltip]", inputs: { tooltipText: ["copilotTooltip", "tooltipText"], tooltipPosition: "tooltipPosition", tooltipDelay: "tooltipDelay" }, host: { listeners: { "mouseenter": "onMouseEnter()", "mouseleave": "onMouseLeave()" } }, ngImport: i0 });
|
|
196
|
+
}
|
|
197
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CopilotTooltip, decorators: [{
|
|
198
|
+
type: Directive,
|
|
199
|
+
args: [{
|
|
200
|
+
selector: "[copilotTooltip]",
|
|
201
|
+
standalone: true,
|
|
202
|
+
}]
|
|
203
|
+
}], propDecorators: { tooltipText: [{
|
|
204
|
+
type: Input,
|
|
205
|
+
args: ["copilotTooltip"]
|
|
206
|
+
}], tooltipPosition: [{
|
|
207
|
+
type: Input
|
|
208
|
+
}], tooltipDelay: [{
|
|
209
|
+
type: Input
|
|
210
|
+
}], onMouseEnter: [{
|
|
211
|
+
type: HostListener,
|
|
212
|
+
args: ["mouseenter"]
|
|
213
|
+
}], onMouseLeave: [{
|
|
214
|
+
type: HostListener,
|
|
215
|
+
args: ["mouseleave"]
|
|
216
|
+
}] } });
|
|
217
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tooltip.js","sourceRoot":"","sources":["../../../../src/lib/directives/tooltip.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Directive,\n  Input,\n  ElementRef,\n  HostListener,\n  OnDestroy,\n  inject,\n  ViewContainerRef,\n} from \"@angular/core\";\nimport {\n  Overlay,\n  OverlayRef,\n  OverlayPositionBuilder,\n  ConnectedPosition,\n} from \"@angular/cdk/overlay\";\nimport { ComponentPortal } from \"@angular/cdk/portal\";\nimport {\n  Component,\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n} from \"@angular/core\";\n\n@Component({\n  selector: \"copilot-tooltip-content\",\n  standalone: true,\n  template: `\n    <div class=\"copilot-tooltip-wrapper\" [attr.data-position]=\"position\">\n      <div class=\"copilot-tooltip\">\n        {{ text }}\n      </div>\n      <div class=\"copilot-tooltip-arrow\"></div>\n    </div>\n  `,\n  styles: [\n    `\n      .copilot-tooltip-wrapper {\n        position: relative;\n        display: inline-block;\n        animation: fadeIn 0.15s ease-in-out;\n      }\n\n      .copilot-tooltip {\n        background-color: #1a1a1a;\n        color: white;\n        padding: 6px 10px;\n        border-radius: 6px;\n        font-size: 12px;\n        font-weight: 500;\n        white-space: nowrap;\n        max-width: 200px;\n        box-shadow: 0 2px 8px rgba(0, 0, 0, 0.15);\n      }\n\n      .copilot-tooltip-arrow {\n        position: absolute;\n        width: 0;\n        height: 0;\n        border-style: solid;\n      }\n\n      /* Arrow for tooltip below element (arrow points up to tooltip) */\n      .copilot-tooltip-wrapper[data-position=\"below\"] .copilot-tooltip-arrow {\n        top: -4px;\n        left: 50%;\n        transform: translateX(-50%);\n        border-width: 0 4px 4px 4px;\n        border-color: transparent transparent #1a1a1a transparent;\n      }\n\n      /* Arrow for tooltip above element (arrow points down to element) */\n      .copilot-tooltip-wrapper[data-position=\"above\"] .copilot-tooltip-arrow {\n        bottom: -4px;\n        left: 50%;\n        transform: translateX(-50%);\n        border-width: 4px 4px 0 4px;\n        border-color: #1a1a1a transparent transparent transparent;\n      }\n\n      /* Arrow for tooltip to the left */\n      .copilot-tooltip-wrapper[data-position=\"left\"] .copilot-tooltip-arrow {\n        right: -4px;\n        top: 50%;\n        transform: translateY(-50%);\n        border-width: 4px 0 4px 4px;\n        border-color: transparent transparent transparent #1a1a1a;\n      }\n\n      /* Arrow for tooltip to the right */\n      .copilot-tooltip-wrapper[data-position=\"right\"] .copilot-tooltip-arrow {\n        left: -4px;\n        top: 50%;\n        transform: translateY(-50%);\n        border-width: 4px 4px 4px 0;\n        border-color: transparent #1a1a1a transparent transparent;\n      }\n\n      @keyframes fadeIn {\n        from {\n          opacity: 0;\n          transform: translateY(2px);\n        }\n        to {\n          opacity: 1;\n          transform: translateY(0);\n        }\n      }\n    `,\n  ],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class TooltipContent {\n  text = \"\";\n  private _position: \"above\" | \"below\" | \"left\" | \"right\" = \"below\";\n\n  get position(): \"above\" | \"below\" | \"left\" | \"right\" {\n    return this._position;\n  }\n\n  set position(value: \"above\" | \"below\" | \"left\" | \"right\") {\n    this._position = value;\n    this.cdr.markForCheck();\n  }\n\n  constructor(private cdr: ChangeDetectorRef) {}\n}\n\n@Directive({\n  selector: \"[copilotTooltip]\",\n  standalone: true,\n})\nexport class CopilotTooltip implements OnDestroy {\n  @Input(\"copilotTooltip\") tooltipText = \"\";\n  @Input() tooltipPosition: \"above\" | \"below\" | \"left\" | \"right\" = \"below\";\n  @Input() tooltipDelay = 500; // milliseconds\n\n  private overlay = inject(Overlay);\n  private overlayPositionBuilder = inject(OverlayPositionBuilder);\n  private elementRef = inject(ElementRef);\n  private viewContainerRef = inject(ViewContainerRef);\n\n  private overlayRef?: OverlayRef;\n  private tooltipTimeout?: number;\n  private originalTitle?: string;\n\n  @HostListener(\"mouseenter\")\n  onMouseEnter(): void {\n    if (!this.tooltipText) return;\n\n    // Store and remove native title to prevent OS tooltip\n    const element = this.elementRef.nativeElement;\n    if (element.hasAttribute(\"title\")) {\n      this.originalTitle = element.getAttribute(\"title\");\n      element.removeAttribute(\"title\");\n    }\n\n    // Clear any existing timeout\n    if (this.tooltipTimeout) {\n      clearTimeout(this.tooltipTimeout);\n    }\n\n    // Set timeout to show tooltip after delay\n    this.tooltipTimeout = window.setTimeout(() => {\n      this.show();\n    }, this.tooltipDelay);\n  }\n\n  @HostListener(\"mouseleave\")\n  onMouseLeave(): void {\n    // Clear timeout if mouse leaves before tooltip shows\n    if (this.tooltipTimeout) {\n      clearTimeout(this.tooltipTimeout);\n      this.tooltipTimeout = undefined;\n    }\n\n    // Restore original title if it existed\n    if (this.originalTitle !== undefined) {\n      this.elementRef.nativeElement.setAttribute(\"title\", this.originalTitle);\n      this.originalTitle = undefined;\n    }\n\n    // Hide tooltip if it's showing\n    this.hide();\n  }\n\n  private show(): void {\n    if (this.overlayRef) {\n      return;\n    }\n\n    // Create overlay\n    const positionStrategy = this.overlayPositionBuilder\n      .flexibleConnectedTo(this.elementRef)\n      .withPositions(this.getPositions())\n      .withPush(false);\n\n    this.overlayRef = this.overlay.create({\n      positionStrategy,\n      scrollStrategy: this.overlay.scrollStrategies.close(),\n      hasBackdrop: false,\n    });\n\n    // Create component portal and attach\n    const portal = new ComponentPortal(TooltipContent, this.viewContainerRef);\n    const componentRef = this.overlayRef.attach(portal);\n    componentRef.instance.text = this.tooltipText;\n\n    // Detect actual position after overlay is positioned\n    setTimeout(() => {\n      if (this.overlayRef && this.elementRef.nativeElement) {\n        const tooltipRect =\n          this.overlayRef.overlayElement.getBoundingClientRect();\n        const elementRect =\n          this.elementRef.nativeElement.getBoundingClientRect();\n\n        let actualPosition: \"above\" | \"below\" | \"left\" | \"right\" = \"below\";\n\n        // Determine actual position based on relative positions\n        if (tooltipRect.bottom <= elementRect.top) {\n          actualPosition = \"above\";\n        } else if (tooltipRect.top >= elementRect.bottom) {\n          actualPosition = \"below\";\n        } else if (tooltipRect.right <= elementRect.left) {\n          actualPosition = \"left\";\n        } else if (tooltipRect.left >= elementRect.right) {\n          actualPosition = \"right\";\n        }\n\n        componentRef.instance.position = actualPosition;\n      }\n    }, 0);\n  }\n\n  private hide(): void {\n    if (this.overlayRef) {\n      this.overlayRef.dispose();\n      this.overlayRef = undefined;\n    }\n  }\n\n  private getPositions(): ConnectedPosition[] {\n    const positions: Record<string, ConnectedPosition[]> = {\n      above: [\n        {\n          originX: \"center\",\n          originY: \"top\",\n          overlayX: \"center\",\n          overlayY: \"bottom\",\n          offsetY: -12,\n        },\n      ],\n      below: [\n        {\n          originX: \"center\",\n          originY: \"bottom\",\n          overlayX: \"center\",\n          overlayY: \"top\",\n          offsetY: 12,\n        },\n      ],\n      left: [\n        {\n          originX: \"start\",\n          originY: \"center\",\n          overlayX: \"end\",\n          overlayY: \"center\",\n          offsetX: -12,\n        },\n      ],\n      right: [\n        {\n          originX: \"end\",\n          originY: \"center\",\n          overlayX: \"start\",\n          overlayY: \"center\",\n          offsetX: 12,\n        },\n      ],\n    };\n\n    // Prefer below position, but use above as fallback\n    const primary = positions[this.tooltipPosition] || positions.below;\n    // For below position, add above as first fallback\n    const fallbacks =\n      this.tooltipPosition === \"below\"\n        ? [\n            ...(positions.above || []),\n            ...(positions.left || []),\n            ...(positions.right || []),\n          ]\n        : Object.values(positions)\n            .filter((p) => p !== primary)\n            .flat();\n\n    return [...(primary || []), ...fallbacks];\n  }\n\n  ngOnDestroy(): void {\n    if (this.tooltipTimeout) {\n      clearTimeout(this.tooltipTimeout);\n    }\n    // Restore original title if it existed\n    if (this.originalTitle !== undefined) {\n      this.elementRef.nativeElement.setAttribute(\"title\", this.originalTitle);\n    }\n    this.hide();\n  }\n}\n"]}
|
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
import { Injectable } from "@angular/core";
|
|
2
|
+
import { filter, lastValueFrom, Subject, take } from "rxjs";
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
export class HumanInTheLoop {
|
|
5
|
+
results = new Subject();
|
|
6
|
+
addResult(toolCallId, toolName, result) {
|
|
7
|
+
this.results.next({ toolCallId, toolName, result });
|
|
8
|
+
}
|
|
9
|
+
onResult(toolCallId, toolName) {
|
|
10
|
+
return lastValueFrom(this.results.pipe(filter((result) => result.toolCallId === toolCallId && result.toolName === toolName), take(1)));
|
|
11
|
+
}
|
|
12
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: HumanInTheLoop, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
|
|
13
|
+
static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: HumanInTheLoop, providedIn: "root" });
|
|
14
|
+
}
|
|
15
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: HumanInTheLoop, decorators: [{
|
|
16
|
+
type: Injectable,
|
|
17
|
+
args: [{ providedIn: "root" }]
|
|
18
|
+
}] });
|
|
19
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaHVtYW4taW4tdGhlLWxvb3AuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvbGliL2h1bWFuLWluLXRoZS1sb29wLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxVQUFVLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDM0MsT0FBTyxFQUFFLE1BQU0sRUFBRSxhQUFhLEVBQUUsT0FBTyxFQUFFLElBQUksRUFBRSxNQUFNLE1BQU0sQ0FBQzs7QUFHNUQsTUFBTSxPQUFPLGNBQWM7SUFDekIsT0FBTyxHQUFHLElBQUksT0FBTyxFQUlqQixDQUFDO0lBRUwsU0FBUyxDQUFDLFVBQWtCLEVBQUUsUUFBZ0IsRUFBRSxNQUFlO1FBQzdELElBQUksQ0FBQyxPQUFPLENBQUMsSUFBSSxDQUFDLEVBQUUsVUFBVSxFQUFFLFFBQVEsRUFBRSxNQUFNLEVBQUUsQ0FBQyxDQUFDO0lBQ3RELENBQUM7SUFFRCxRQUFRLENBQUMsVUFBa0IsRUFBRSxRQUFnQjtRQUMzQyxPQUFPLGFBQWEsQ0FDbEIsSUFBSSxDQUFDLE9BQU8sQ0FBQyxJQUFJLENBQ2YsTUFBTSxDQUNKLENBQUMsTUFBTSxFQUFFLEVBQUUsQ0FDVCxNQUFNLENBQUMsVUFBVSxLQUFLLFVBQVUsSUFBSSxNQUFNLENBQUMsUUFBUSxLQUFLLFFBQVEsQ0FDbkUsRUFDRCxJQUFJLENBQUMsQ0FBQyxDQUFDLENBQ1IsQ0FDRixDQUFDO0lBQ0osQ0FBQzt3R0FyQlUsY0FBYzs0R0FBZCxjQUFjLGNBREQsTUFBTTs7NEZBQ25CLGNBQWM7a0JBRDFCLFVBQVU7bUJBQUMsRUFBRSxVQUFVLEVBQUUsTUFBTSxFQUFFIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgSW5qZWN0YWJsZSB9IGZyb20gXCJAYW5ndWxhci9jb3JlXCI7XG5pbXBvcnQgeyBmaWx0ZXIsIGxhc3RWYWx1ZUZyb20sIFN1YmplY3QsIHRha2UgfSBmcm9tIFwicnhqc1wiO1xuXG5ASW5qZWN0YWJsZSh7IHByb3ZpZGVkSW46IFwicm9vdFwiIH0pXG5leHBvcnQgY2xhc3MgSHVtYW5JblRoZUxvb3Age1xuICByZXN1bHRzID0gbmV3IFN1YmplY3Q8e1xuICAgIHRvb2xDYWxsSWQ6IHN0cmluZztcbiAgICB0b29sTmFtZTogc3RyaW5nO1xuICAgIHJlc3VsdDogdW5rbm93bjtcbiAgfT4oKTtcblxuICBhZGRSZXN1bHQodG9vbENhbGxJZDogc3RyaW5nLCB0b29sTmFtZTogc3RyaW5nLCByZXN1bHQ6IHVua25vd24pIHtcbiAgICB0aGlzLnJlc3VsdHMubmV4dCh7IHRvb2xDYWxsSWQsIHRvb2xOYW1lLCByZXN1bHQgfSk7XG4gIH1cblxuICBvblJlc3VsdCh0b29sQ2FsbElkOiBzdHJpbmcsIHRvb2xOYW1lOiBzdHJpbmcpOiBQcm9taXNlPHVua25vd24+IHtcbiAgICByZXR1cm4gbGFzdFZhbHVlRnJvbShcbiAgICAgIHRoaXMucmVzdWx0cy5waXBlKFxuICAgICAgICBmaWx0ZXIoXG4gICAgICAgICAgKHJlc3VsdCkgPT5cbiAgICAgICAgICAgIHJlc3VsdC50b29sQ2FsbElkID09PSB0b29sQ2FsbElkICYmIHJlc3VsdC50b29sTmFtZSA9PT0gdG9vbE5hbWVcbiAgICAgICAgKSxcbiAgICAgICAgdGFrZSgxKVxuICAgICAgKVxuICAgICk7XG4gIH1cbn1cbiJdfQ==
|
|
@@ -0,0 +1,131 @@
|
|
|
1
|
+
import { NgComponentOutlet } from "@angular/common";
|
|
2
|
+
import { Component, inject, input } from "@angular/core";
|
|
3
|
+
import { CopilotKit } from "./copilotkit";
|
|
4
|
+
import { partialJSONParse } from "@copilotkitnext/shared";
|
|
5
|
+
import { HumanInTheLoop } from "./human-in-the-loop";
|
|
6
|
+
import * as i0 from "@angular/core";
|
|
7
|
+
export class RenderToolCalls {
|
|
8
|
+
#copilotKit = inject(CopilotKit);
|
|
9
|
+
#hitl = inject(HumanInTheLoop);
|
|
10
|
+
message = input.required();
|
|
11
|
+
messages = input.required();
|
|
12
|
+
isLoading = input(false);
|
|
13
|
+
pickRenderer(name) {
|
|
14
|
+
const messageAgentId = this.message()
|
|
15
|
+
.agentId;
|
|
16
|
+
const renderers = this.#copilotKit.toolCallRenderConfigs();
|
|
17
|
+
const clientTools = this.#copilotKit.clientToolCallRenderConfigs();
|
|
18
|
+
const humanInTheLoopTools = this.#copilotKit.humanInTheLoopToolRenderConfigs();
|
|
19
|
+
const renderer = renderers.find((candidate) => candidate.name === name &&
|
|
20
|
+
(candidate.agentId === undefined ||
|
|
21
|
+
candidate.agentId === messageAgentId));
|
|
22
|
+
if (renderer)
|
|
23
|
+
return { type: "renderer", config: renderer };
|
|
24
|
+
const clientTool = clientTools.find((candidate) => candidate.name === name &&
|
|
25
|
+
(candidate.agentId === undefined ||
|
|
26
|
+
candidate.agentId === messageAgentId));
|
|
27
|
+
if (clientTool)
|
|
28
|
+
return { type: "clientTool", config: clientTool };
|
|
29
|
+
const humanInTheLoopTool = humanInTheLoopTools.find((candidate) => candidate.name === name &&
|
|
30
|
+
(candidate.agentId === undefined ||
|
|
31
|
+
candidate.agentId === messageAgentId));
|
|
32
|
+
if (humanInTheLoopTool)
|
|
33
|
+
return { type: "humanInTheLoopTool", config: humanInTheLoopTool };
|
|
34
|
+
const starRenderer = renderers.find((candidate) => candidate.name === "*");
|
|
35
|
+
if (starRenderer)
|
|
36
|
+
return { type: "renderer", config: starRenderer };
|
|
37
|
+
return undefined;
|
|
38
|
+
}
|
|
39
|
+
buildToolCall(toolCall) {
|
|
40
|
+
const args = partialJSONParse(toolCall.function.arguments);
|
|
41
|
+
const message = this.#getToolMessage(toolCall.id);
|
|
42
|
+
if (message) {
|
|
43
|
+
return {
|
|
44
|
+
args,
|
|
45
|
+
status: "complete",
|
|
46
|
+
result: message.content,
|
|
47
|
+
};
|
|
48
|
+
}
|
|
49
|
+
else if (this.isLoading()) {
|
|
50
|
+
return {
|
|
51
|
+
args,
|
|
52
|
+
status: "in-progress",
|
|
53
|
+
result: undefined,
|
|
54
|
+
};
|
|
55
|
+
}
|
|
56
|
+
else {
|
|
57
|
+
return {
|
|
58
|
+
args,
|
|
59
|
+
status: "executing",
|
|
60
|
+
result: undefined,
|
|
61
|
+
};
|
|
62
|
+
}
|
|
63
|
+
}
|
|
64
|
+
buildHumanInTheLoopToolCall(toolCall) {
|
|
65
|
+
const args = partialJSONParse(toolCall.function.arguments);
|
|
66
|
+
const message = this.#getToolMessage(toolCall.id);
|
|
67
|
+
if (message) {
|
|
68
|
+
return {
|
|
69
|
+
args,
|
|
70
|
+
status: "complete",
|
|
71
|
+
result: message.content,
|
|
72
|
+
};
|
|
73
|
+
}
|
|
74
|
+
else if (this.isLoading()) {
|
|
75
|
+
return {
|
|
76
|
+
args,
|
|
77
|
+
status: "in-progress",
|
|
78
|
+
result: undefined,
|
|
79
|
+
};
|
|
80
|
+
}
|
|
81
|
+
else {
|
|
82
|
+
return {
|
|
83
|
+
args,
|
|
84
|
+
status: "executing",
|
|
85
|
+
result: undefined,
|
|
86
|
+
respond: (result) => {
|
|
87
|
+
this.#hitl.addResult(toolCall.id, toolCall.function.name, result);
|
|
88
|
+
},
|
|
89
|
+
};
|
|
90
|
+
}
|
|
91
|
+
}
|
|
92
|
+
#getToolMessage(toolCallId) {
|
|
93
|
+
return this.messages().find((m) => m.role === "tool" && m.toolCallId === toolCallId);
|
|
94
|
+
}
|
|
95
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: RenderToolCalls, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
96
|
+
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: RenderToolCalls, isStandalone: true, selector: "copilot-render-tool-calls", inputs: { message: { classPropertyName: "message", publicName: "message", isSignal: true, isRequired: true, transformFunction: null }, messages: { classPropertyName: "messages", publicName: "messages", isSignal: true, isRequired: true, transformFunction: null }, isLoading: { classPropertyName: "isLoading", publicName: "isLoading", isSignal: true, isRequired: false, transformFunction: null } }, ngImport: i0, template: `
|
|
97
|
+
@for (toolCall of message().toolCalls ?? []; track toolCall.id) {
|
|
98
|
+
@let renderConfig = pickRenderer(toolCall.function.name);
|
|
99
|
+
@if (renderConfig && renderConfig.type !== "humanInTheLoopTool") {
|
|
100
|
+
<ng-container
|
|
101
|
+
*ngComponentOutlet="
|
|
102
|
+
renderConfig.config.component;
|
|
103
|
+
inputs: { toolCall: buildToolCall(toolCall) }
|
|
104
|
+
"
|
|
105
|
+
/>
|
|
106
|
+
}
|
|
107
|
+
}
|
|
108
|
+
`, isInline: true, dependencies: [{ kind: "directive", type: NgComponentOutlet, selector: "[ngComponentOutlet]", inputs: ["ngComponentOutlet", "ngComponentOutletInputs", "ngComponentOutletInjector", "ngComponentOutletContent", "ngComponentOutletNgModule", "ngComponentOutletNgModuleFactory"] }] });
|
|
109
|
+
}
|
|
110
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: RenderToolCalls, decorators: [{
|
|
111
|
+
type: Component,
|
|
112
|
+
args: [{
|
|
113
|
+
selector: "copilot-render-tool-calls",
|
|
114
|
+
standalone: true,
|
|
115
|
+
imports: [NgComponentOutlet],
|
|
116
|
+
template: `
|
|
117
|
+
@for (toolCall of message().toolCalls ?? []; track toolCall.id) {
|
|
118
|
+
@let renderConfig = pickRenderer(toolCall.function.name);
|
|
119
|
+
@if (renderConfig && renderConfig.type !== "humanInTheLoopTool") {
|
|
120
|
+
<ng-container
|
|
121
|
+
*ngComponentOutlet="
|
|
122
|
+
renderConfig.config.component;
|
|
123
|
+
inputs: { toolCall: buildToolCall(toolCall) }
|
|
124
|
+
"
|
|
125
|
+
/>
|
|
126
|
+
}
|
|
127
|
+
}
|
|
128
|
+
`,
|
|
129
|
+
}]
|
|
130
|
+
}] });
|
|
131
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"render-tool-calls.js","sourceRoot":"","sources":["../../../src/lib/render-tool-calls.ts"],"names":[],"mappings":"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","sourcesContent":["import { NgComponentOutlet } from \"@angular/common\";\nimport { Component, inject, input } from \"@angular/core\";\nimport { AssistantMessage, Message, ToolCall } from \"@ag-ui/client\";\nimport { CopilotKit } from \"./copilotkit\";\nimport {\n  FrontendToolConfig,\n  HumanInTheLoopToolCall,\n  HumanInTheLoopConfig,\n  AngularToolCall,\n  RenderToolCallConfig,\n} from \"./tools\";\nimport { partialJSONParse } from \"@copilotkitnext/shared\";\nimport { HumanInTheLoop } from \"./human-in-the-loop\";\n\ntype RendererToolCallHandler = {\n  type: \"renderer\";\n  config: RenderToolCallConfig;\n};\ntype ClientToolCallHandler = {\n  type: \"clientTool\";\n  config: FrontendToolConfig;\n};\ntype HumanInTheLoopToolCallHandler = {\n  type: \"humanInTheLoopTool\";\n  config: HumanInTheLoopConfig;\n};\n\ntype ToolCallHandler =\n  | RendererToolCallHandler\n  | ClientToolCallHandler\n  | HumanInTheLoopToolCallHandler;\n\n@Component({\n  selector: \"copilot-render-tool-calls\",\n  standalone: true,\n  imports: [NgComponentOutlet],\n  template: `\n    @for (toolCall of message().toolCalls ?? []; track toolCall.id) {\n      @let renderConfig = pickRenderer(toolCall.function.name);\n      @if (renderConfig && renderConfig.type !== \"humanInTheLoopTool\") {\n        <ng-container\n          *ngComponentOutlet=\"\n            renderConfig.config.component;\n            inputs: { toolCall: buildToolCall(toolCall) }\n          \"\n        />\n      }\n    }\n  `,\n})\nexport class RenderToolCalls {\n  readonly #copilotKit = inject(CopilotKit);\n  readonly #hitl = inject(HumanInTheLoop);\n  readonly message = input.required<AssistantMessage>();\n  readonly messages = input.required<Message[]>();\n  readonly isLoading = input<boolean>(false);\n\n  protected pickRenderer(name: string): ToolCallHandler | undefined {\n    type AssistantMessageWithAgent = AssistantMessage & {\n      agentId?: string;\n    };\n    const messageAgentId = (this.message() as AssistantMessageWithAgent)\n      .agentId;\n    const renderers = this.#copilotKit.toolCallRenderConfigs();\n    const clientTools = this.#copilotKit.clientToolCallRenderConfigs();\n    const humanInTheLoopTools =\n      this.#copilotKit.humanInTheLoopToolRenderConfigs();\n\n    const renderer = renderers.find(\n      (candidate) =>\n        candidate.name === name &&\n        (candidate.agentId === undefined ||\n          candidate.agentId === messageAgentId)\n    );\n\n    if (renderer) return { type: \"renderer\", config: renderer };\n\n    const clientTool = clientTools.find(\n      (candidate) =>\n        candidate.name === name &&\n        (candidate.agentId === undefined ||\n          candidate.agentId === messageAgentId)\n    );\n    if (clientTool) return { type: \"clientTool\", config: clientTool };\n\n    const humanInTheLoopTool = humanInTheLoopTools.find(\n      (candidate) =>\n        candidate.name === name &&\n        (candidate.agentId === undefined ||\n          candidate.agentId === messageAgentId)\n    );\n    if (humanInTheLoopTool)\n      return { type: \"humanInTheLoopTool\", config: humanInTheLoopTool };\n\n    const starRenderer = renderers.find((candidate) => candidate.name === \"*\");\n    if (starRenderer) return { type: \"renderer\", config: starRenderer };\n\n    return undefined;\n  }\n\n  protected buildToolCall<Args extends Record<string, unknown>>(\n    toolCall: ToolCall\n  ): AngularToolCall<Args> {\n    const args = partialJSONParse(toolCall.function.arguments);\n    const message = this.#getToolMessage(toolCall.id);\n\n    if (message) {\n      return {\n        args,\n        status: \"complete\",\n        result: message.content!,\n      };\n    } else if (this.isLoading()) {\n      return {\n        args,\n        status: \"in-progress\",\n        result: undefined,\n      };\n    } else {\n      return {\n        args,\n        status: \"executing\",\n        result: undefined,\n      };\n    }\n  }\n\n  protected buildHumanInTheLoopToolCall<Args extends Record<string, unknown>>(\n    toolCall: ToolCall\n  ): HumanInTheLoopToolCall<Args> {\n    const args = partialJSONParse(toolCall.function.arguments);\n    const message = this.#getToolMessage(toolCall.id);\n\n    if (message) {\n      return {\n        args,\n        status: \"complete\",\n        result: message.content!,\n      };\n    } else if (this.isLoading()) {\n      return {\n        args,\n        status: \"in-progress\",\n        result: undefined,\n      };\n    } else {\n      return {\n        args,\n        status: \"executing\",\n        result: undefined,\n        respond: (result) => {\n          this.#hitl.addResult(toolCall.id, toolCall.function.name, result);\n        },\n      };\n    }\n  }\n\n  #getToolMessage(toolCallId: string): Message | undefined {\n    return this.messages().find(\n      (m) => m.role === \"tool\" && m.toolCallId === toolCallId\n    );\n  }\n}\n"]}
|