@copilotkitnext/angular 0.0.1 → 0.0.4
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +248 -0
- package/dist/README.md +248 -0
- package/dist/components/chat/copilot-chat-assistant-message.component.d.ts +10 -10
- package/dist/components/chat/copilot-chat-message-view.component.d.ts +42 -42
- package/dist/components/chat/copilot-chat-view.component.d.ts +14 -14
- package/dist/esm2022/components/chat/copilot-chat-assistant-message-buttons.component.mjs +384 -0
- package/dist/esm2022/components/chat/copilot-chat-assistant-message-renderer.component.mjs +286 -0
- package/dist/esm2022/components/chat/copilot-chat-assistant-message-toolbar.component.mjs +27 -0
- package/dist/esm2022/components/chat/copilot-chat-assistant-message.component.mjs +433 -0
- package/dist/esm2022/components/chat/copilot-chat-assistant-message.types.mjs +2 -0
- package/dist/esm2022/components/chat/copilot-chat-audio-recorder.component.mjs +202 -0
- package/dist/esm2022/components/chat/copilot-chat-buttons.component.mjs +321 -0
- package/dist/esm2022/components/chat/copilot-chat-input-defaults.mjs +38 -0
- package/dist/esm2022/components/chat/copilot-chat-input.component.mjs +666 -0
- package/dist/esm2022/components/chat/copilot-chat-input.types.mjs +10 -0
- package/dist/esm2022/components/chat/copilot-chat-message-view-cursor.component.mjs +45 -0
- package/dist/esm2022/components/chat/copilot-chat-message-view.component.mjs +296 -0
- package/dist/esm2022/components/chat/copilot-chat-message-view.types.mjs +2 -0
- package/dist/esm2022/components/chat/copilot-chat-textarea.component.mjs +188 -0
- package/dist/esm2022/components/chat/copilot-chat-tool-calls-view.component.mjs +216 -0
- package/dist/esm2022/components/chat/copilot-chat-toolbar.component.mjs +25 -0
- package/dist/esm2022/components/chat/copilot-chat-tools-menu.component.mjs +199 -0
- package/dist/esm2022/components/chat/copilot-chat-user-message-branch-navigation.component.mjs +137 -0
- package/dist/esm2022/components/chat/copilot-chat-user-message-buttons.component.mjs +207 -0
- package/dist/esm2022/components/chat/copilot-chat-user-message-renderer.component.mjs +35 -0
- package/dist/esm2022/components/chat/copilot-chat-user-message-toolbar.component.mjs +34 -0
- package/dist/esm2022/components/chat/copilot-chat-user-message.component.mjs +341 -0
- package/dist/esm2022/components/chat/copilot-chat-user-message.types.mjs +2 -0
- package/dist/esm2022/components/chat/copilot-chat-view-disclaimer.component.mjs +52 -0
- package/dist/esm2022/components/chat/copilot-chat-view-feather.component.mjs +55 -0
- package/dist/esm2022/components/chat/copilot-chat-view-handlers.service.mjs +19 -0
- package/dist/esm2022/components/chat/copilot-chat-view-input-container.component.mjs +110 -0
- package/dist/esm2022/components/chat/copilot-chat-view-scroll-to-bottom-button.component.mjs +93 -0
- package/dist/esm2022/components/chat/copilot-chat-view-scroll-view.component.mjs +443 -0
- package/dist/esm2022/components/chat/copilot-chat-view.component.mjs +479 -0
- package/dist/esm2022/components/chat/copilot-chat-view.types.mjs +2 -0
- package/dist/esm2022/components/chat/copilot-chat.component.mjs +214 -0
- package/dist/esm2022/components/copilotkit-tool-render.component.mjs +153 -0
- package/dist/esm2022/copilotkitnext-angular.mjs +5 -0
- package/dist/esm2022/core/chat-configuration/chat-configuration.providers.mjs +65 -0
- package/dist/esm2022/core/chat-configuration/chat-configuration.service.mjs +145 -0
- package/dist/esm2022/core/chat-configuration/chat-configuration.types.mjs +26 -0
- package/dist/esm2022/core/copilotkit.providers.mjs +34 -0
- package/dist/esm2022/core/copilotkit.service.mjs +430 -0
- package/dist/esm2022/core/copilotkit.types.mjs +12 -0
- package/dist/esm2022/directives/copilotkit-agent-context.directive.mjs +130 -0
- package/dist/esm2022/directives/copilotkit-agent.directive.mjs +217 -0
- package/dist/esm2022/directives/copilotkit-chat-config.directive.mjs +218 -0
- package/dist/esm2022/directives/copilotkit-config.directive.mjs +94 -0
- package/dist/esm2022/directives/copilotkit-frontend-tool.directive.mjs +130 -0
- package/dist/esm2022/directives/copilotkit-human-in-the-loop.directive.mjs +266 -0
- package/dist/esm2022/directives/stick-to-bottom.directive.mjs +181 -0
- package/dist/esm2022/index.mjs +70 -0
- package/dist/esm2022/lib/directives/tooltip.directive.mjs +211 -0
- package/dist/esm2022/lib/slots/copilot-slot.component.mjs +144 -0
- package/dist/esm2022/lib/slots/slot.types.mjs +6 -0
- package/dist/esm2022/lib/slots/slot.utils.mjs +222 -0
- package/dist/esm2022/lib/utils.mjs +10 -0
- package/dist/esm2022/services/resize-observer.service.mjs +152 -0
- package/dist/esm2022/services/scroll-position.service.mjs +124 -0
- package/dist/esm2022/types/frontend-tool.mjs +2 -0
- package/dist/esm2022/types/human-in-the-loop.mjs +2 -0
- package/dist/esm2022/utils/agent-context.utils.mjs +114 -0
- package/dist/esm2022/utils/agent.utils.mjs +204 -0
- package/dist/esm2022/utils/chat-config.utils.mjs +186 -0
- package/dist/esm2022/utils/copilotkit.utils.mjs +20 -0
- package/dist/esm2022/utils/frontend-tool.utils.mjs +228 -0
- package/dist/esm2022/utils/human-in-the-loop.utils.mjs +296 -0
- package/dist/fesm2022/copilotkitnext-angular.mjs +163 -164
- package/dist/fesm2022/copilotkitnext-angular.mjs.map +1 -1
- package/dist/styles.css +0 -27
- package/package.json +23 -20
- package/vitest.config.mts +32 -21
- package/.turbo/turbo-build.log +0 -39
- package/.turbo/turbo-check-types.log +0 -0
- package/.turbo/turbo-test.log +0 -71
- package/README-agent-context.md +0 -310
- package/ng-package.json +0 -19
- package/slots.md +0 -331
- package/src/components/chat/__tests__/copilot-chat-assistant-message.component.spec.ts +0 -282
- package/src/components/chat/__tests__/copilot-chat-input.component.spec.ts +0 -419
- package/src/components/chat/__tests__/copilot-chat-message-view.component.spec.ts +0 -372
- package/src/components/chat/__tests__/copilot-chat-user-message.component.spec.ts +0 -249
- package/src/components/chat/copilot-chat-assistant-message-buttons.component.ts +0 -292
- package/src/components/chat/copilot-chat-assistant-message-renderer.component.ts +0 -472
- package/src/components/chat/copilot-chat-assistant-message-toolbar.component.ts +0 -29
- package/src/components/chat/copilot-chat-assistant-message.component.ts +0 -463
- package/src/components/chat/copilot-chat-assistant-message.types.ts +0 -50
- package/src/components/chat/copilot-chat-audio-recorder.component.ts +0 -241
- package/src/components/chat/copilot-chat-buttons.component.ts +0 -308
- package/src/components/chat/copilot-chat-buttons.component.ts.bak +0 -471
- package/src/components/chat/copilot-chat-input-defaults.ts +0 -47
- package/src/components/chat/copilot-chat-input.component.ts +0 -512
- package/src/components/chat/copilot-chat-input.types.ts +0 -148
- package/src/components/chat/copilot-chat-message-view-cursor.component.ts +0 -51
- package/src/components/chat/copilot-chat-message-view.component.ts +0 -233
- package/src/components/chat/copilot-chat-message-view.types.ts +0 -39
- package/src/components/chat/copilot-chat-textarea.component.ts +0 -220
- package/src/components/chat/copilot-chat-tool-calls-view.component.ts +0 -261
- package/src/components/chat/copilot-chat-toolbar.component.ts +0 -35
- package/src/components/chat/copilot-chat-tools-menu.component.ts +0 -185
- package/src/components/chat/copilot-chat-user-message-branch-navigation.component.ts +0 -121
- package/src/components/chat/copilot-chat-user-message-buttons.component.ts +0 -170
- package/src/components/chat/copilot-chat-user-message-renderer.component.ts +0 -37
- package/src/components/chat/copilot-chat-user-message-toolbar.component.ts +0 -37
- package/src/components/chat/copilot-chat-user-message.component.ts +0 -247
- package/src/components/chat/copilot-chat-user-message.types.ts +0 -42
- package/src/components/chat/copilot-chat-view-disclaimer.component.ts +0 -51
- package/src/components/chat/copilot-chat-view-feather.component.ts +0 -47
- package/src/components/chat/copilot-chat-view-handlers.service.ts +0 -14
- package/src/components/chat/copilot-chat-view-input-container.component.ts +0 -87
- package/src/components/chat/copilot-chat-view-scroll-to-bottom-button.component.ts +0 -79
- package/src/components/chat/copilot-chat-view-scroll-view.component.ts +0 -322
- package/src/components/chat/copilot-chat-view.component.ts +0 -420
- package/src/components/chat/copilot-chat-view.types.ts +0 -52
- package/src/components/chat/copilot-chat.component.ts +0 -232
- package/src/components/copilotkit-tool-render.component.ts +0 -169
- package/src/core/__tests__/copilotkit.service.spec.ts +0 -1051
- package/src/core/__tests__/copilotkit.service.wildcard.spec.ts +0 -316
- package/src/core/chat-configuration/__tests__/chat-configuration.service.spec.ts +0 -287
- package/src/core/chat-configuration/chat-configuration.providers.ts +0 -71
- package/src/core/chat-configuration/chat-configuration.service.ts +0 -162
- package/src/core/chat-configuration/chat-configuration.types.ts +0 -57
- package/src/core/copilotkit.providers.ts +0 -59
- package/src/core/copilotkit.service.ts +0 -542
- package/src/core/copilotkit.types.ts +0 -132
- package/src/directives/__tests__/copilotkit-agent-context.directive.spec.ts +0 -384
- package/src/directives/__tests__/copilotkit-agent.directive.spec.ts +0 -253
- package/src/directives/__tests__/copilotkit-chat-config.directive.spec.ts +0 -385
- package/src/directives/__tests__/copilotkit-config.directive.spec.ts +0 -69
- package/src/directives/__tests__/copilotkit-frontend-tool-simple.directive.spec.ts +0 -60
- package/src/directives/__tests__/copilotkit-frontend-tool.directive.spec.ts +0 -108
- package/src/directives/__tests__/copilotkit-human-in-the-loop.directive.spec.ts +0 -452
- package/src/directives/copilotkit-agent-context.directive.ts +0 -138
- package/src/directives/copilotkit-agent.directive.ts +0 -225
- package/src/directives/copilotkit-chat-config.directive.ts +0 -241
- package/src/directives/copilotkit-config.directive.ts +0 -81
- package/src/directives/copilotkit-frontend-tool.directive.ts +0 -145
- package/src/directives/copilotkit-human-in-the-loop.directive.ts +0 -281
- package/src/directives/stick-to-bottom.directive.ts +0 -204
- package/src/index.ts +0 -105
- package/src/lib/directives/tooltip.directive.ts +0 -292
- package/src/lib/slots/__tests__/slot.utils.spec.ts +0 -377
- package/src/lib/slots/copilot-slot.component.ts +0 -135
- package/src/lib/slots/index.ts +0 -3
- package/src/lib/slots/slot.types.ts +0 -64
- package/src/lib/slots/slot.utils.ts +0 -289
- package/src/lib/utils.ts +0 -10
- package/src/public-api.ts +0 -1
- package/src/services/resize-observer.service.ts +0 -181
- package/src/services/scroll-position.service.ts +0 -169
- package/src/styles/globals.css +0 -266
- package/src/styles/index.css +0 -3
- package/src/test-setup.ts +0 -15
- package/src/testing/index.ts +0 -3
- package/src/testing/testing.utils.ts +0 -248
- package/src/types/frontend-tool.ts +0 -44
- package/src/types/human-in-the-loop.ts +0 -52
- package/src/utils/__tests__/agent.utils.spec.ts +0 -234
- package/src/utils/__tests__/chat-config.utils.spec.ts +0 -306
- package/src/utils/__tests__/frontend-tool-inject.spec.ts +0 -350
- package/src/utils/__tests__/frontend-tool-integration.spec.ts +0 -199
- package/src/utils/__tests__/frontend-tool.utils.spec.ts +0 -272
- package/src/utils/__tests__/human-in-the-loop.utils.spec.ts +0 -365
- package/src/utils/agent-context.utils.ts +0 -133
- package/src/utils/agent.utils.ts +0 -239
- package/src/utils/chat-config.utils.ts +0 -221
- package/src/utils/copilotkit.utils.ts +0 -20
- package/src/utils/frontend-tool.utils.ts +0 -266
- package/src/utils/human-in-the-loop.utils.ts +0 -359
- package/tsconfig.spec.json +0 -12
|
@@ -0,0 +1,181 @@
|
|
|
1
|
+
import { Directive, Input, Output, EventEmitter, ElementRef, inject } from '@angular/core';
|
|
2
|
+
import { ScrollPositionService } from '../services/scroll-position.service';
|
|
3
|
+
import { ResizeObserverService } from '../services/resize-observer.service';
|
|
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 StickToBottomDirective {
|
|
24
|
+
enabled = true;
|
|
25
|
+
threshold = 10;
|
|
26
|
+
initialBehavior = 'smooth';
|
|
27
|
+
resizeBehavior = 'smooth';
|
|
28
|
+
debounceMs = 100;
|
|
29
|
+
isAtBottomChange = new EventEmitter();
|
|
30
|
+
scrollToBottomRequested = new EventEmitter();
|
|
31
|
+
elementRef = inject(ElementRef);
|
|
32
|
+
scrollService = inject(ScrollPositionService);
|
|
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.monitorScrollPosition(element, this.threshold)
|
|
66
|
+
.pipe(takeUntil(this.destroy$), debounceTime(this.debounceMs), distinctUntilChanged((a, b) => a.isAtBottom === b.isAtBottom))
|
|
67
|
+
.subscribe(state => {
|
|
68
|
+
const wasAtBottom = this.wasAtBottom;
|
|
69
|
+
this.wasAtBottom = state.isAtBottom;
|
|
70
|
+
// Detect user scroll
|
|
71
|
+
if (!state.isAtBottom && wasAtBottom && this.hasInitialized) {
|
|
72
|
+
this.userHasScrolled = true;
|
|
73
|
+
}
|
|
74
|
+
else if (state.isAtBottom) {
|
|
75
|
+
this.userHasScrolled = false;
|
|
76
|
+
}
|
|
77
|
+
// Emit change
|
|
78
|
+
this.isAtBottomChange.emit(state.isAtBottom);
|
|
79
|
+
});
|
|
80
|
+
}
|
|
81
|
+
setupResizeMonitoring() {
|
|
82
|
+
if (!this.enabled || !this.contentElement)
|
|
83
|
+
return;
|
|
84
|
+
// Monitor content resize
|
|
85
|
+
this.resizeService.observeElement(this.contentElement, 0, 250)
|
|
86
|
+
.pipe(takeUntil(this.destroy$), filter(() => this.enabled && !this.userHasScrolled))
|
|
87
|
+
.subscribe(state => {
|
|
88
|
+
// Auto-scroll on resize if we were at bottom
|
|
89
|
+
if (this.wasAtBottom && !state.isResizing) {
|
|
90
|
+
this.scrollToBottom(this.resizeBehavior);
|
|
91
|
+
}
|
|
92
|
+
});
|
|
93
|
+
// Monitor container resize
|
|
94
|
+
const element = this.elementRef.nativeElement;
|
|
95
|
+
this.resizeService.observeElement(element, 0, 250)
|
|
96
|
+
.pipe(takeUntil(this.destroy$), filter(() => this.enabled && !this.userHasScrolled && this.wasAtBottom))
|
|
97
|
+
.subscribe(() => {
|
|
98
|
+
// Adjust scroll on container resize
|
|
99
|
+
this.scrollToBottom(this.resizeBehavior);
|
|
100
|
+
});
|
|
101
|
+
}
|
|
102
|
+
setupContentMutationObserver() {
|
|
103
|
+
if (!this.enabled || !this.contentElement)
|
|
104
|
+
return;
|
|
105
|
+
const mutationObserver = new MutationObserver(() => {
|
|
106
|
+
if (this.enabled && this.wasAtBottom && !this.userHasScrolled) {
|
|
107
|
+
// Content changed, scroll to bottom if we were there
|
|
108
|
+
requestAnimationFrame(() => {
|
|
109
|
+
this.scrollToBottom(this.resizeBehavior);
|
|
110
|
+
});
|
|
111
|
+
}
|
|
112
|
+
});
|
|
113
|
+
mutationObserver.observe(this.contentElement, {
|
|
114
|
+
childList: true,
|
|
115
|
+
subtree: true,
|
|
116
|
+
characterData: true
|
|
117
|
+
});
|
|
118
|
+
// Cleanup on destroy
|
|
119
|
+
this.destroy$.subscribe(() => {
|
|
120
|
+
mutationObserver.disconnect();
|
|
121
|
+
});
|
|
122
|
+
}
|
|
123
|
+
/**
|
|
124
|
+
* Public method to scroll to bottom
|
|
125
|
+
* Can be called from parent component
|
|
126
|
+
*/
|
|
127
|
+
scrollToBottom(behavior = 'smooth') {
|
|
128
|
+
const element = this.elementRef.nativeElement;
|
|
129
|
+
const smooth = behavior === 'smooth';
|
|
130
|
+
this.scrollService.scrollToBottom(element, smooth);
|
|
131
|
+
this.userHasScrolled = false;
|
|
132
|
+
this.scrollToBottomRequested.emit();
|
|
133
|
+
}
|
|
134
|
+
/**
|
|
135
|
+
* Check if currently at bottom
|
|
136
|
+
*/
|
|
137
|
+
isAtBottom() {
|
|
138
|
+
return this.scrollService.isAtBottom(this.elementRef.nativeElement, this.threshold);
|
|
139
|
+
}
|
|
140
|
+
/**
|
|
141
|
+
* Get current scroll state
|
|
142
|
+
*/
|
|
143
|
+
getScrollState() {
|
|
144
|
+
const element = this.elementRef.nativeElement;
|
|
145
|
+
return {
|
|
146
|
+
scrollTop: element.scrollTop,
|
|
147
|
+
scrollHeight: element.scrollHeight,
|
|
148
|
+
clientHeight: element.clientHeight,
|
|
149
|
+
isAtBottom: this.isAtBottom()
|
|
150
|
+
};
|
|
151
|
+
}
|
|
152
|
+
ngOnDestroy() {
|
|
153
|
+
this.destroy$.next();
|
|
154
|
+
this.destroy$.complete();
|
|
155
|
+
}
|
|
156
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: StickToBottomDirective, deps: [], target: i0.ɵɵFactoryTarget.Directive });
|
|
157
|
+
static ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "18.2.13", type: StickToBottomDirective, isStandalone: true, selector: "[copilotStickToBottom]", inputs: { enabled: "enabled", threshold: "threshold", initialBehavior: "initialBehavior", resizeBehavior: "resizeBehavior", debounceMs: "debounceMs" }, outputs: { isAtBottomChange: "isAtBottomChange", scrollToBottomRequested: "scrollToBottomRequested" }, providers: [ScrollPositionService, ResizeObserverService], ngImport: i0 });
|
|
158
|
+
}
|
|
159
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: StickToBottomDirective, decorators: [{
|
|
160
|
+
type: Directive,
|
|
161
|
+
args: [{
|
|
162
|
+
selector: '[copilotStickToBottom]',
|
|
163
|
+
standalone: true,
|
|
164
|
+
providers: [ScrollPositionService, ResizeObserverService]
|
|
165
|
+
}]
|
|
166
|
+
}], propDecorators: { enabled: [{
|
|
167
|
+
type: Input
|
|
168
|
+
}], threshold: [{
|
|
169
|
+
type: Input
|
|
170
|
+
}], initialBehavior: [{
|
|
171
|
+
type: Input
|
|
172
|
+
}], resizeBehavior: [{
|
|
173
|
+
type: Input
|
|
174
|
+
}], debounceMs: [{
|
|
175
|
+
type: Input
|
|
176
|
+
}], isAtBottomChange: [{
|
|
177
|
+
type: Output
|
|
178
|
+
}], scrollToBottomRequested: [{
|
|
179
|
+
type: Output
|
|
180
|
+
}] } });
|
|
181
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"stick-to-bottom.directive.js","sourceRoot":"","sources":["../../../src/directives/stick-to-bottom.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Directive,\n  Input,\n  Output,\n  EventEmitter,\n  ElementRef,\n  OnInit,\n  OnDestroy,\n  AfterViewInit,\n  inject\n} from '@angular/core';\nimport { ScrollPositionService } from '../services/scroll-position.service';\nimport { ResizeObserverService } from '../services/resize-observer.service';\nimport { Subject, combineLatest, merge } from 'rxjs';\nimport { takeUntil, debounceTime, filter, distinctUntilChanged } 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  selector: '[copilotStickToBottom]',\n  standalone: true,\n  providers: [ScrollPositionService, ResizeObserverService]\n})\nexport class StickToBottomDirective implements OnInit, AfterViewInit, OnDestroy {\n  @Input() enabled: boolean = true;\n  @Input() threshold: number = 10;\n  @Input() initialBehavior: ScrollBehavior = 'smooth';\n  @Input() resizeBehavior: ScrollBehavior = 'smooth';\n  @Input() debounceMs: number = 100;\n  \n  @Output() isAtBottomChange = new EventEmitter<boolean>();\n  @Output() scrollToBottomRequested = new EventEmitter<void>();\n  \n  private elementRef = inject(ElementRef);\n  private scrollService = inject(ScrollPositionService);\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('[data-stick-to-bottom-content]') 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.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.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.observeElement(element, 0, 250)\n      .pipe(\n        takeUntil(this.destroy$),\n        filter(() => this.enabled && !this.userHasScrolled && this.wasAtBottom)\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(this.elementRef.nativeElement, this.threshold);\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}"]}
|
|
@@ -0,0 +1,70 @@
|
|
|
1
|
+
export * from "./core/copilotkit.service";
|
|
2
|
+
export * from "./core/copilotkit.types";
|
|
3
|
+
export * from "./core/copilotkit.providers";
|
|
4
|
+
export * from "./core/chat-configuration/chat-configuration.types";
|
|
5
|
+
export * from "./core/chat-configuration/chat-configuration.service";
|
|
6
|
+
export * from "./core/chat-configuration/chat-configuration.providers";
|
|
7
|
+
export * from "./utils/copilotkit.utils";
|
|
8
|
+
export * from "./utils/agent-context.utils";
|
|
9
|
+
export * from "./utils/frontend-tool.utils";
|
|
10
|
+
// Note: tool-render.utils removed in favor of direct ToolCallRender<T> usage
|
|
11
|
+
// Export all except AgentWatchResult which is already exported from copilotkit.types
|
|
12
|
+
export { watchAgent, getAgent, subscribeToAgent, registerAgentWatcher, } from "./utils/agent.utils";
|
|
13
|
+
export * from "./utils/human-in-the-loop.utils";
|
|
14
|
+
export * from "./utils/chat-config.utils";
|
|
15
|
+
// Slot utilities are internal only, not exported
|
|
16
|
+
// export * from "./lib/slots/slot.types";
|
|
17
|
+
// export * from "./lib/slots/slot.utils";
|
|
18
|
+
// export { CopilotSlotComponent } from "./lib/slots/copilot-slot.component";
|
|
19
|
+
export { CopilotTooltipDirective } from "./lib/directives/tooltip.directive";
|
|
20
|
+
export { CopilotKitConfigDirective } from "./directives/copilotkit-config.directive";
|
|
21
|
+
export { CopilotKitAgentContextDirective } from "./directives/copilotkit-agent-context.directive";
|
|
22
|
+
export { CopilotKitFrontendToolDirective } from "./directives/copilotkit-frontend-tool.directive";
|
|
23
|
+
export { CopilotKitAgentDirective } from "./directives/copilotkit-agent.directive";
|
|
24
|
+
export { CopilotKitHumanInTheLoopDirective, CopilotKitHumanInTheLoopRespondDirective, } from "./directives/copilotkit-human-in-the-loop.directive";
|
|
25
|
+
export { CopilotKitChatConfigDirective } from "./directives/copilotkit-chat-config.directive";
|
|
26
|
+
export { CopilotKitToolRenderComponent } from "./components/copilotkit-tool-render.component";
|
|
27
|
+
// Chat Input Components
|
|
28
|
+
export * from "./components/chat/copilot-chat-input.types";
|
|
29
|
+
export { CopilotChatInputComponent } from "./components/chat/copilot-chat-input.component";
|
|
30
|
+
export { CopilotChatInputDefaults } from "./components/chat/copilot-chat-input-defaults";
|
|
31
|
+
export { CopilotChatTextareaComponent } from "./components/chat/copilot-chat-textarea.component";
|
|
32
|
+
export { CopilotChatAudioRecorderComponent } from "./components/chat/copilot-chat-audio-recorder.component";
|
|
33
|
+
export { CopilotChatSendButtonComponent, CopilotChatToolbarButtonComponent, CopilotChatStartTranscribeButtonComponent, CopilotChatCancelTranscribeButtonComponent, CopilotChatFinishTranscribeButtonComponent, CopilotChatAddFileButtonComponent, } from "./components/chat/copilot-chat-buttons.component";
|
|
34
|
+
export { CopilotChatToolbarComponent } from "./components/chat/copilot-chat-toolbar.component";
|
|
35
|
+
export { CopilotChatToolsMenuComponent } from "./components/chat/copilot-chat-tools-menu.component";
|
|
36
|
+
// Chat User Message Components
|
|
37
|
+
export * from "./components/chat/copilot-chat-user-message.types";
|
|
38
|
+
export { CopilotChatUserMessageComponent } from "./components/chat/copilot-chat-user-message.component";
|
|
39
|
+
export { CopilotChatUserMessageRendererComponent } from "./components/chat/copilot-chat-user-message-renderer.component";
|
|
40
|
+
export { CopilotChatUserMessageToolbarButtonComponent, CopilotChatUserMessageCopyButtonComponent, CopilotChatUserMessageEditButtonComponent, } from "./components/chat/copilot-chat-user-message-buttons.component";
|
|
41
|
+
export { CopilotChatUserMessageToolbarComponent } from "./components/chat/copilot-chat-user-message-toolbar.component";
|
|
42
|
+
export { CopilotChatUserMessageBranchNavigationComponent } from "./components/chat/copilot-chat-user-message-branch-navigation.component";
|
|
43
|
+
// Chat Assistant Message Components
|
|
44
|
+
export * from "./components/chat/copilot-chat-assistant-message.types";
|
|
45
|
+
export { CopilotChatAssistantMessageComponent } from "./components/chat/copilot-chat-assistant-message.component";
|
|
46
|
+
export { CopilotChatAssistantMessageRendererComponent } from "./components/chat/copilot-chat-assistant-message-renderer.component";
|
|
47
|
+
export { CopilotChatAssistantMessageToolbarButtonComponent, CopilotChatAssistantMessageCopyButtonComponent, CopilotChatAssistantMessageThumbsUpButtonComponent, CopilotChatAssistantMessageThumbsDownButtonComponent, CopilotChatAssistantMessageReadAloudButtonComponent, CopilotChatAssistantMessageRegenerateButtonComponent, } from "./components/chat/copilot-chat-assistant-message-buttons.component";
|
|
48
|
+
export { CopilotChatAssistantMessageToolbarComponent } from "./components/chat/copilot-chat-assistant-message-toolbar.component";
|
|
49
|
+
// Chat Message View Components
|
|
50
|
+
export * from "./components/chat/copilot-chat-message-view.types";
|
|
51
|
+
export { CopilotChatMessageViewComponent } from "./components/chat/copilot-chat-message-view.component";
|
|
52
|
+
export { CopilotChatMessageViewCursorComponent } from "./components/chat/copilot-chat-message-view-cursor.component";
|
|
53
|
+
export { CopilotChatToolCallsViewComponent } from "./components/chat/copilot-chat-tool-calls-view.component";
|
|
54
|
+
// Chat View Components
|
|
55
|
+
export * from "./components/chat/copilot-chat-view.types";
|
|
56
|
+
export { CopilotChatViewComponent } from "./components/chat/copilot-chat-view.component";
|
|
57
|
+
export { CopilotChatViewScrollViewComponent } from "./components/chat/copilot-chat-view-scroll-view.component";
|
|
58
|
+
export { CopilotChatViewScrollToBottomButtonComponent } from "./components/chat/copilot-chat-view-scroll-to-bottom-button.component";
|
|
59
|
+
export { CopilotChatViewFeatherComponent } from "./components/chat/copilot-chat-view-feather.component";
|
|
60
|
+
export { CopilotChatViewInputContainerComponent } from "./components/chat/copilot-chat-view-input-container.component";
|
|
61
|
+
export { CopilotChatViewDisclaimerComponent } from "./components/chat/copilot-chat-view-disclaimer.component";
|
|
62
|
+
// Main Chat Component
|
|
63
|
+
export { CopilotChatComponent } from "./components/chat/copilot-chat.component";
|
|
64
|
+
// Services and Directives for Chat View
|
|
65
|
+
export { ScrollPositionService } from "./services/scroll-position.service";
|
|
66
|
+
export { ResizeObserverService } from "./services/resize-observer.service";
|
|
67
|
+
export { StickToBottomDirective } from "./directives/stick-to-bottom.directive";
|
|
68
|
+
// Testing utilities are not exported from the main entry point
|
|
69
|
+
// They should be imported directly from '@copilotkitnext/angular/testing' if needed
|
|
70
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/index.ts"],"names":[],"mappings":"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","sourcesContent":["export * from \"./core/copilotkit.service\";\nexport * from \"./core/copilotkit.types\";\nexport * from \"./core/copilotkit.providers\";\n// Re-export types from @ag-ui/core for convenience\nexport type { Message, ToolCall, ToolMessage } from \"@ag-ui/core\";\nexport * from \"./core/chat-configuration/chat-configuration.types\";\nexport * from \"./core/chat-configuration/chat-configuration.service\";\nexport * from \"./core/chat-configuration/chat-configuration.providers\";\nexport * from \"./utils/copilotkit.utils\";\nexport * from \"./utils/agent-context.utils\";\nexport * from \"./utils/frontend-tool.utils\";\n// Note: tool-render.utils removed in favor of direct ToolCallRender<T> usage\n// Export all except AgentWatchResult which is already exported from copilotkit.types\nexport {\n  watchAgent,\n  getAgent,\n  subscribeToAgent,\n  registerAgentWatcher,\n} from \"./utils/agent.utils\";\nexport * from \"./utils/human-in-the-loop.utils\";\nexport * from \"./utils/chat-config.utils\";\n// Slot utilities are internal only, not exported\n// export * from \"./lib/slots/slot.types\";\n// export * from \"./lib/slots/slot.utils\";\n// export { CopilotSlotComponent } from \"./lib/slots/copilot-slot.component\";\nexport { CopilotTooltipDirective } from \"./lib/directives/tooltip.directive\";\nexport { CopilotKitConfigDirective } from \"./directives/copilotkit-config.directive\";\nexport { CopilotKitAgentContextDirective } from \"./directives/copilotkit-agent-context.directive\";\nexport { CopilotKitFrontendToolDirective } from \"./directives/copilotkit-frontend-tool.directive\";\nexport { CopilotKitAgentDirective } from \"./directives/copilotkit-agent.directive\";\nexport {\n  CopilotKitHumanInTheLoopDirective,\n  CopilotKitHumanInTheLoopRespondDirective,\n} from \"./directives/copilotkit-human-in-the-loop.directive\";\nexport { CopilotKitChatConfigDirective } from \"./directives/copilotkit-chat-config.directive\";\nexport { CopilotKitToolRenderComponent } from \"./components/copilotkit-tool-render.component\";\n\n// Chat Input Components\nexport * from \"./components/chat/copilot-chat-input.types\";\nexport { CopilotChatInputComponent } from \"./components/chat/copilot-chat-input.component\";\nexport { CopilotChatInputDefaults } from \"./components/chat/copilot-chat-input-defaults\";\nexport { CopilotChatTextareaComponent } from \"./components/chat/copilot-chat-textarea.component\";\nexport { CopilotChatAudioRecorderComponent } from \"./components/chat/copilot-chat-audio-recorder.component\";\nexport {\n  CopilotChatSendButtonComponent,\n  CopilotChatToolbarButtonComponent,\n  CopilotChatStartTranscribeButtonComponent,\n  CopilotChatCancelTranscribeButtonComponent,\n  CopilotChatFinishTranscribeButtonComponent,\n  CopilotChatAddFileButtonComponent,\n} from \"./components/chat/copilot-chat-buttons.component\";\nexport { CopilotChatToolbarComponent } from \"./components/chat/copilot-chat-toolbar.component\";\nexport { CopilotChatToolsMenuComponent } from \"./components/chat/copilot-chat-tools-menu.component\";\n\n// Chat User Message Components\nexport * from \"./components/chat/copilot-chat-user-message.types\";\nexport { CopilotChatUserMessageComponent } from \"./components/chat/copilot-chat-user-message.component\";\nexport { CopilotChatUserMessageRendererComponent } from \"./components/chat/copilot-chat-user-message-renderer.component\";\nexport {\n  CopilotChatUserMessageToolbarButtonComponent,\n  CopilotChatUserMessageCopyButtonComponent,\n  CopilotChatUserMessageEditButtonComponent,\n} from \"./components/chat/copilot-chat-user-message-buttons.component\";\nexport { CopilotChatUserMessageToolbarComponent } from \"./components/chat/copilot-chat-user-message-toolbar.component\";\nexport { CopilotChatUserMessageBranchNavigationComponent } from \"./components/chat/copilot-chat-user-message-branch-navigation.component\";\n\n// Chat Assistant Message Components\nexport * from \"./components/chat/copilot-chat-assistant-message.types\";\nexport { CopilotChatAssistantMessageComponent } from \"./components/chat/copilot-chat-assistant-message.component\";\nexport { CopilotChatAssistantMessageRendererComponent } from \"./components/chat/copilot-chat-assistant-message-renderer.component\";\nexport {\n  CopilotChatAssistantMessageToolbarButtonComponent,\n  CopilotChatAssistantMessageCopyButtonComponent,\n  CopilotChatAssistantMessageThumbsUpButtonComponent,\n  CopilotChatAssistantMessageThumbsDownButtonComponent,\n  CopilotChatAssistantMessageReadAloudButtonComponent,\n  CopilotChatAssistantMessageRegenerateButtonComponent,\n} from \"./components/chat/copilot-chat-assistant-message-buttons.component\";\nexport { CopilotChatAssistantMessageToolbarComponent } from \"./components/chat/copilot-chat-assistant-message-toolbar.component\";\n\n// Chat Message View Components\nexport * from \"./components/chat/copilot-chat-message-view.types\";\nexport { CopilotChatMessageViewComponent } from \"./components/chat/copilot-chat-message-view.component\";\nexport { CopilotChatMessageViewCursorComponent } from \"./components/chat/copilot-chat-message-view-cursor.component\";\nexport { CopilotChatToolCallsViewComponent } from \"./components/chat/copilot-chat-tool-calls-view.component\";\n\n// Chat View Components\nexport * from \"./components/chat/copilot-chat-view.types\";\nexport { CopilotChatViewComponent } from \"./components/chat/copilot-chat-view.component\";\nexport { CopilotChatViewScrollViewComponent } from \"./components/chat/copilot-chat-view-scroll-view.component\";\nexport { CopilotChatViewScrollToBottomButtonComponent } from \"./components/chat/copilot-chat-view-scroll-to-bottom-button.component\";\nexport { CopilotChatViewFeatherComponent } from \"./components/chat/copilot-chat-view-feather.component\";\nexport { CopilotChatViewInputContainerComponent } from \"./components/chat/copilot-chat-view-input-container.component\";\nexport { CopilotChatViewDisclaimerComponent } from \"./components/chat/copilot-chat-view-disclaimer.component\";\n\n// Main Chat Component\nexport { CopilotChatComponent } from \"./components/chat/copilot-chat.component\";\n\n// Services and Directives for Chat View\nexport { ScrollPositionService } from \"./services/scroll-position.service\";\nexport { ResizeObserverService } from \"./services/resize-observer.service\";\nexport { StickToBottomDirective } from \"./directives/stick-to-bottom.directive\";\n\n// Testing utilities are not exported from the main entry point\n// They should be imported directly from '@copilotkitnext/angular/testing' if needed\n"]}
|
|
@@ -0,0 +1,211 @@
|
|
|
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 TooltipContentComponent {
|
|
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: TooltipContentComponent, deps: [{ token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
|
|
21
|
+
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: TooltipContentComponent, 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: TooltipContentComponent, decorators: [{
|
|
31
|
+
type: Component,
|
|
32
|
+
args: [{ selector: 'copilot-tooltip-content', 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, standalone: 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"] }]
|
|
40
|
+
}], ctorParameters: () => [{ type: i0.ChangeDetectorRef }] });
|
|
41
|
+
export class CopilotTooltipDirective {
|
|
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(TooltipContentComponent, 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
|
+
? [...(positions.above || []), ...(positions.left || []), ...(positions.right || [])]
|
|
175
|
+
: Object.values(positions).filter(p => p !== primary).flat();
|
|
176
|
+
return [...(primary || []), ...fallbacks];
|
|
177
|
+
}
|
|
178
|
+
ngOnDestroy() {
|
|
179
|
+
if (this.tooltipTimeout) {
|
|
180
|
+
clearTimeout(this.tooltipTimeout);
|
|
181
|
+
}
|
|
182
|
+
// Restore original title if it existed
|
|
183
|
+
if (this.originalTitle !== undefined) {
|
|
184
|
+
this.elementRef.nativeElement.setAttribute('title', this.originalTitle);
|
|
185
|
+
}
|
|
186
|
+
this.hide();
|
|
187
|
+
}
|
|
188
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CopilotTooltipDirective, deps: [], target: i0.ɵɵFactoryTarget.Directive });
|
|
189
|
+
static ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "18.2.13", type: CopilotTooltipDirective, isStandalone: true, selector: "[copilotTooltip]", inputs: { tooltipText: ["copilotTooltip", "tooltipText"], tooltipPosition: "tooltipPosition", tooltipDelay: "tooltipDelay" }, host: { listeners: { "mouseenter": "onMouseEnter()", "mouseleave": "onMouseLeave()" } }, ngImport: i0 });
|
|
190
|
+
}
|
|
191
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CopilotTooltipDirective, decorators: [{
|
|
192
|
+
type: Directive,
|
|
193
|
+
args: [{
|
|
194
|
+
selector: '[copilotTooltip]',
|
|
195
|
+
standalone: true
|
|
196
|
+
}]
|
|
197
|
+
}], propDecorators: { tooltipText: [{
|
|
198
|
+
type: Input,
|
|
199
|
+
args: ['copilotTooltip']
|
|
200
|
+
}], tooltipPosition: [{
|
|
201
|
+
type: Input
|
|
202
|
+
}], tooltipDelay: [{
|
|
203
|
+
type: Input
|
|
204
|
+
}], onMouseEnter: [{
|
|
205
|
+
type: HostListener,
|
|
206
|
+
args: ['mouseenter']
|
|
207
|
+
}], onMouseLeave: [{
|
|
208
|
+
type: HostListener,
|
|
209
|
+
args: ['mouseleave']
|
|
210
|
+
}] } });
|
|
211
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tooltip.directive.js","sourceRoot":"","sources":["../../../../src/lib/directives/tooltip.directive.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 { Component, ChangeDetectionStrategy, ChangeDetectorRef } from '@angular/core';\n\n@Component({\n  selector: 'copilot-tooltip-content',\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    .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  changeDetection: ChangeDetectionStrategy.OnPush,\n  standalone: true\n})\nexport class TooltipContentComponent {\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 CopilotTooltipDirective 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(TooltipContentComponent, 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 = this.overlayRef.overlayElement.getBoundingClientRect();\n        const elementRect = 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 = this.tooltipPosition === 'below' \n      ? [...(positions.above || []), ...(positions.left || []), ...(positions.right || [])]\n      : Object.values(positions).filter(p => p !== primary).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}"]}
|