@copilotkitnext/angular 0.0.2 → 0.0.5

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.
Files changed (173) hide show
  1. package/README.md +3 -3
  2. package/dist/README.md +3 -3
  3. package/dist/components/chat/copilot-chat-assistant-message.component.d.ts +10 -10
  4. package/dist/components/chat/copilot-chat-message-view.component.d.ts +42 -42
  5. package/dist/components/chat/copilot-chat-view.component.d.ts +14 -14
  6. package/dist/core/copilotkit.providers.d.ts +1 -1
  7. package/dist/core/copilotkit.service.d.ts +5 -5
  8. package/dist/core/copilotkit.types.d.ts +8 -10
  9. package/dist/directives/copilotkit-frontend-tool.directive.d.ts +1 -1
  10. package/dist/esm2022/components/chat/copilot-chat-assistant-message-buttons.component.mjs +384 -0
  11. package/dist/esm2022/components/chat/copilot-chat-assistant-message-renderer.component.mjs +286 -0
  12. package/dist/esm2022/components/chat/copilot-chat-assistant-message-toolbar.component.mjs +27 -0
  13. package/dist/esm2022/components/chat/copilot-chat-assistant-message.component.mjs +433 -0
  14. package/dist/esm2022/components/chat/copilot-chat-assistant-message.types.mjs +2 -0
  15. package/dist/esm2022/components/chat/copilot-chat-audio-recorder.component.mjs +202 -0
  16. package/dist/esm2022/components/chat/copilot-chat-buttons.component.mjs +321 -0
  17. package/dist/esm2022/components/chat/copilot-chat-input-defaults.mjs +38 -0
  18. package/dist/esm2022/components/chat/copilot-chat-input.component.mjs +666 -0
  19. package/dist/esm2022/components/chat/copilot-chat-input.types.mjs +10 -0
  20. package/dist/esm2022/components/chat/copilot-chat-message-view-cursor.component.mjs +45 -0
  21. package/dist/esm2022/components/chat/copilot-chat-message-view.component.mjs +296 -0
  22. package/dist/esm2022/components/chat/copilot-chat-message-view.types.mjs +2 -0
  23. package/dist/esm2022/components/chat/copilot-chat-textarea.component.mjs +188 -0
  24. package/dist/esm2022/components/chat/copilot-chat-tool-calls-view.component.mjs +216 -0
  25. package/dist/esm2022/components/chat/copilot-chat-toolbar.component.mjs +25 -0
  26. package/dist/esm2022/components/chat/copilot-chat-tools-menu.component.mjs +199 -0
  27. package/dist/esm2022/components/chat/copilot-chat-user-message-branch-navigation.component.mjs +137 -0
  28. package/dist/esm2022/components/chat/copilot-chat-user-message-buttons.component.mjs +207 -0
  29. package/dist/esm2022/components/chat/copilot-chat-user-message-renderer.component.mjs +35 -0
  30. package/dist/esm2022/components/chat/copilot-chat-user-message-toolbar.component.mjs +34 -0
  31. package/dist/esm2022/components/chat/copilot-chat-user-message.component.mjs +341 -0
  32. package/dist/esm2022/components/chat/copilot-chat-user-message.types.mjs +2 -0
  33. package/dist/esm2022/components/chat/copilot-chat-view-disclaimer.component.mjs +52 -0
  34. package/dist/esm2022/components/chat/copilot-chat-view-feather.component.mjs +55 -0
  35. package/dist/esm2022/components/chat/copilot-chat-view-handlers.service.mjs +19 -0
  36. package/dist/esm2022/components/chat/copilot-chat-view-input-container.component.mjs +110 -0
  37. package/dist/esm2022/components/chat/copilot-chat-view-scroll-to-bottom-button.component.mjs +93 -0
  38. package/dist/esm2022/components/chat/copilot-chat-view-scroll-view.component.mjs +443 -0
  39. package/dist/esm2022/components/chat/copilot-chat-view.component.mjs +479 -0
  40. package/dist/esm2022/components/chat/copilot-chat-view.types.mjs +2 -0
  41. package/dist/esm2022/components/chat/copilot-chat.component.mjs +214 -0
  42. package/dist/esm2022/components/copilotkit-tool-render.component.mjs +153 -0
  43. package/dist/esm2022/copilotkitnext-angular.mjs +5 -0
  44. package/dist/esm2022/core/chat-configuration/chat-configuration.providers.mjs +65 -0
  45. package/dist/esm2022/core/chat-configuration/chat-configuration.service.mjs +145 -0
  46. package/dist/esm2022/core/chat-configuration/chat-configuration.types.mjs +26 -0
  47. package/dist/esm2022/core/copilotkit.providers.mjs +34 -0
  48. package/dist/esm2022/core/copilotkit.service.mjs +426 -0
  49. package/dist/esm2022/core/copilotkit.types.mjs +13 -0
  50. package/dist/esm2022/directives/copilotkit-agent-context.directive.mjs +130 -0
  51. package/dist/esm2022/directives/copilotkit-agent.directive.mjs +217 -0
  52. package/dist/esm2022/directives/copilotkit-chat-config.directive.mjs +218 -0
  53. package/dist/esm2022/directives/copilotkit-config.directive.mjs +94 -0
  54. package/dist/esm2022/directives/copilotkit-frontend-tool.directive.mjs +128 -0
  55. package/dist/esm2022/directives/copilotkit-human-in-the-loop.directive.mjs +265 -0
  56. package/dist/esm2022/directives/stick-to-bottom.directive.mjs +181 -0
  57. package/dist/esm2022/index.mjs +70 -0
  58. package/dist/esm2022/lib/directives/tooltip.directive.mjs +211 -0
  59. package/dist/esm2022/lib/slots/copilot-slot.component.mjs +144 -0
  60. package/dist/esm2022/lib/slots/slot.types.mjs +6 -0
  61. package/dist/esm2022/lib/slots/slot.utils.mjs +222 -0
  62. package/dist/esm2022/lib/utils.mjs +10 -0
  63. package/dist/esm2022/services/resize-observer.service.mjs +152 -0
  64. package/dist/esm2022/services/scroll-position.service.mjs +124 -0
  65. package/dist/esm2022/types/frontend-tool.mjs +2 -0
  66. package/dist/esm2022/types/human-in-the-loop.mjs +2 -0
  67. package/dist/esm2022/utils/agent-context.utils.mjs +114 -0
  68. package/dist/esm2022/utils/agent.utils.mjs +204 -0
  69. package/dist/esm2022/utils/chat-config.utils.mjs +186 -0
  70. package/dist/esm2022/utils/copilotkit.utils.mjs +20 -0
  71. package/dist/esm2022/utils/frontend-tool.utils.mjs +224 -0
  72. package/dist/esm2022/utils/human-in-the-loop.utils.mjs +293 -0
  73. package/dist/fesm2022/copilotkitnext-angular.mjs +174 -187
  74. package/dist/fesm2022/copilotkitnext-angular.mjs.map +1 -1
  75. package/dist/utils/frontend-tool.utils.d.ts +1 -1
  76. package/package.json +23 -20
  77. package/vitest.config.mts +32 -21
  78. package/.turbo/turbo-build.log +0 -38
  79. package/.turbo/turbo-check-types.log +0 -0
  80. package/.turbo/turbo-test.log +0 -71
  81. package/ng-package.json +0 -19
  82. package/src/components/chat/__tests__/copilot-chat-assistant-message.component.spec.ts +0 -282
  83. package/src/components/chat/__tests__/copilot-chat-input.component.spec.ts +0 -419
  84. package/src/components/chat/__tests__/copilot-chat-message-view.component.spec.ts +0 -372
  85. package/src/components/chat/__tests__/copilot-chat-user-message.component.spec.ts +0 -249
  86. package/src/components/chat/copilot-chat-assistant-message-buttons.component.ts +0 -292
  87. package/src/components/chat/copilot-chat-assistant-message-renderer.component.ts +0 -472
  88. package/src/components/chat/copilot-chat-assistant-message-toolbar.component.ts +0 -29
  89. package/src/components/chat/copilot-chat-assistant-message.component.ts +0 -463
  90. package/src/components/chat/copilot-chat-assistant-message.types.ts +0 -50
  91. package/src/components/chat/copilot-chat-audio-recorder.component.ts +0 -241
  92. package/src/components/chat/copilot-chat-buttons.component.ts +0 -308
  93. package/src/components/chat/copilot-chat-buttons.component.ts.bak +0 -471
  94. package/src/components/chat/copilot-chat-input-defaults.ts +0 -47
  95. package/src/components/chat/copilot-chat-input.component.ts +0 -512
  96. package/src/components/chat/copilot-chat-input.types.ts +0 -148
  97. package/src/components/chat/copilot-chat-message-view-cursor.component.ts +0 -51
  98. package/src/components/chat/copilot-chat-message-view.component.ts +0 -233
  99. package/src/components/chat/copilot-chat-message-view.types.ts +0 -39
  100. package/src/components/chat/copilot-chat-textarea.component.ts +0 -220
  101. package/src/components/chat/copilot-chat-tool-calls-view.component.ts +0 -261
  102. package/src/components/chat/copilot-chat-toolbar.component.ts +0 -35
  103. package/src/components/chat/copilot-chat-tools-menu.component.ts +0 -185
  104. package/src/components/chat/copilot-chat-user-message-branch-navigation.component.ts +0 -121
  105. package/src/components/chat/copilot-chat-user-message-buttons.component.ts +0 -170
  106. package/src/components/chat/copilot-chat-user-message-renderer.component.ts +0 -37
  107. package/src/components/chat/copilot-chat-user-message-toolbar.component.ts +0 -37
  108. package/src/components/chat/copilot-chat-user-message.component.ts +0 -247
  109. package/src/components/chat/copilot-chat-user-message.types.ts +0 -42
  110. package/src/components/chat/copilot-chat-view-disclaimer.component.ts +0 -51
  111. package/src/components/chat/copilot-chat-view-feather.component.ts +0 -47
  112. package/src/components/chat/copilot-chat-view-handlers.service.ts +0 -14
  113. package/src/components/chat/copilot-chat-view-input-container.component.ts +0 -87
  114. package/src/components/chat/copilot-chat-view-scroll-to-bottom-button.component.ts +0 -79
  115. package/src/components/chat/copilot-chat-view-scroll-view.component.ts +0 -322
  116. package/src/components/chat/copilot-chat-view.component.ts +0 -420
  117. package/src/components/chat/copilot-chat-view.types.ts +0 -52
  118. package/src/components/chat/copilot-chat.component.ts +0 -232
  119. package/src/components/copilotkit-tool-render.component.ts +0 -169
  120. package/src/core/__tests__/copilotkit.service.spec.ts +0 -1051
  121. package/src/core/__tests__/copilotkit.service.wildcard.spec.ts +0 -316
  122. package/src/core/chat-configuration/__tests__/chat-configuration.service.spec.ts +0 -287
  123. package/src/core/chat-configuration/chat-configuration.providers.ts +0 -71
  124. package/src/core/chat-configuration/chat-configuration.service.ts +0 -162
  125. package/src/core/chat-configuration/chat-configuration.types.ts +0 -57
  126. package/src/core/copilotkit.providers.ts +0 -59
  127. package/src/core/copilotkit.service.ts +0 -542
  128. package/src/core/copilotkit.types.ts +0 -132
  129. package/src/directives/__tests__/copilotkit-agent-context.directive.spec.ts +0 -384
  130. package/src/directives/__tests__/copilotkit-agent.directive.spec.ts +0 -253
  131. package/src/directives/__tests__/copilotkit-chat-config.directive.spec.ts +0 -385
  132. package/src/directives/__tests__/copilotkit-config.directive.spec.ts +0 -69
  133. package/src/directives/__tests__/copilotkit-frontend-tool-simple.directive.spec.ts +0 -60
  134. package/src/directives/__tests__/copilotkit-frontend-tool.directive.spec.ts +0 -108
  135. package/src/directives/__tests__/copilotkit-human-in-the-loop.directive.spec.ts +0 -452
  136. package/src/directives/copilotkit-agent-context.directive.ts +0 -138
  137. package/src/directives/copilotkit-agent.directive.ts +0 -225
  138. package/src/directives/copilotkit-chat-config.directive.ts +0 -241
  139. package/src/directives/copilotkit-config.directive.ts +0 -81
  140. package/src/directives/copilotkit-frontend-tool.directive.ts +0 -145
  141. package/src/directives/copilotkit-human-in-the-loop.directive.ts +0 -281
  142. package/src/directives/stick-to-bottom.directive.ts +0 -204
  143. package/src/index.ts +0 -105
  144. package/src/lib/directives/tooltip.directive.ts +0 -292
  145. package/src/lib/slots/__tests__/slot.utils.spec.ts +0 -377
  146. package/src/lib/slots/copilot-slot.component.ts +0 -135
  147. package/src/lib/slots/index.ts +0 -3
  148. package/src/lib/slots/slot.types.ts +0 -64
  149. package/src/lib/slots/slot.utils.ts +0 -289
  150. package/src/lib/utils.ts +0 -10
  151. package/src/public-api.ts +0 -1
  152. package/src/services/resize-observer.service.ts +0 -181
  153. package/src/services/scroll-position.service.ts +0 -169
  154. package/src/styles/globals.css +0 -266
  155. package/src/styles/index.css +0 -3
  156. package/src/test-setup.ts +0 -15
  157. package/src/testing/index.ts +0 -3
  158. package/src/testing/testing.utils.ts +0 -248
  159. package/src/types/frontend-tool.ts +0 -44
  160. package/src/types/human-in-the-loop.ts +0 -52
  161. package/src/utils/__tests__/agent.utils.spec.ts +0 -234
  162. package/src/utils/__tests__/chat-config.utils.spec.ts +0 -306
  163. package/src/utils/__tests__/frontend-tool-inject.spec.ts +0 -350
  164. package/src/utils/__tests__/frontend-tool-integration.spec.ts +0 -199
  165. package/src/utils/__tests__/frontend-tool.utils.spec.ts +0 -272
  166. package/src/utils/__tests__/human-in-the-loop.utils.spec.ts +0 -365
  167. package/src/utils/agent-context.utils.ts +0 -133
  168. package/src/utils/agent.utils.ts +0 -239
  169. package/src/utils/chat-config.utils.ts +0 -221
  170. package/src/utils/copilotkit.utils.ts +0 -20
  171. package/src/utils/frontend-tool.utils.ts +0 -266
  172. package/src/utils/human-in-the-loop.utils.ts +0 -359
  173. package/tsconfig.spec.json +0 -12
@@ -0,0 +1,93 @@
1
+ import { Component, Input, Output, EventEmitter, ChangeDetectionStrategy, ViewEncapsulation } from '@angular/core';
2
+ import { CommonModule } from '@angular/common';
3
+ import { LucideAngularModule, ChevronDown } from 'lucide-angular';
4
+ import { cn } from '../../lib/utils';
5
+ import * as i0 from "@angular/core";
6
+ import * as i1 from "lucide-angular";
7
+ /**
8
+ * ScrollToBottomButton component for CopilotChatView
9
+ * Matches React implementation exactly with same Tailwind classes
10
+ */
11
+ export class CopilotChatViewScrollToBottomButtonComponent {
12
+ inputClass;
13
+ disabled = false;
14
+ // Support function-style click handler via slot context
15
+ onClick;
16
+ // Simple, idiomatic Angular output
17
+ clicked = new EventEmitter();
18
+ // Icon reference
19
+ ChevronDown = ChevronDown;
20
+ // Computed class matching React exactly
21
+ get computedClass() {
22
+ return cn(
23
+ // Base button styles
24
+ 'rounded-full w-10 h-10 p-0',
25
+ // Background colors
26
+ 'bg-white dark:bg-gray-900',
27
+ // Border and shadow
28
+ 'shadow-lg border border-gray-200 dark:border-gray-700',
29
+ // Hover states
30
+ 'hover:bg-gray-50 dark:hover:bg-gray-800',
31
+ // Layout
32
+ 'flex items-center justify-center cursor-pointer',
33
+ // Transition
34
+ 'transition-colors',
35
+ // Focus states
36
+ 'focus:outline-none focus-visible:ring-2 focus-visible:ring-offset-2',
37
+ // Custom classes
38
+ this.inputClass);
39
+ }
40
+ handleClick() {
41
+ if (!this.disabled) {
42
+ // Call input handler if provided (slot-style)
43
+ if (this.onClick) {
44
+ this.onClick();
45
+ }
46
+ this.clicked.emit();
47
+ }
48
+ }
49
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CopilotChatViewScrollToBottomButtonComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
50
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: CopilotChatViewScrollToBottomButtonComponent, isStandalone: true, selector: "copilot-chat-view-scroll-to-bottom-button", inputs: { inputClass: "inputClass", disabled: "disabled", onClick: "onClick" }, outputs: { clicked: "clicked" }, ngImport: i0, template: `
51
+ <button
52
+ type="button"
53
+ [class]="computedClass"
54
+ [disabled]="disabled"
55
+ (click)="handleClick()">
56
+ <lucide-angular
57
+ [img]="ChevronDown"
58
+ class="w-4 h-4 text-gray-600 dark:text-white">
59
+ </lucide-angular>
60
+ </button>
61
+ `, isInline: true, dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "ngmodule", type: LucideAngularModule }, { kind: "component", type: i1.LucideAngularComponent, selector: "lucide-angular, lucide-icon, i-lucide, span-lucide", inputs: ["class", "name", "img", "color", "absoluteStrokeWidth", "size", "strokeWidth"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None });
62
+ }
63
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CopilotChatViewScrollToBottomButtonComponent, decorators: [{
64
+ type: Component,
65
+ args: [{
66
+ selector: 'copilot-chat-view-scroll-to-bottom-button',
67
+ standalone: true,
68
+ imports: [CommonModule, LucideAngularModule],
69
+ changeDetection: ChangeDetectionStrategy.OnPush,
70
+ encapsulation: ViewEncapsulation.None,
71
+ template: `
72
+ <button
73
+ type="button"
74
+ [class]="computedClass"
75
+ [disabled]="disabled"
76
+ (click)="handleClick()">
77
+ <lucide-angular
78
+ [img]="ChevronDown"
79
+ class="w-4 h-4 text-gray-600 dark:text-white">
80
+ </lucide-angular>
81
+ </button>
82
+ `
83
+ }]
84
+ }], propDecorators: { inputClass: [{
85
+ type: Input
86
+ }], disabled: [{
87
+ type: Input
88
+ }], onClick: [{
89
+ type: Input
90
+ }], clicked: [{
91
+ type: Output
92
+ }] } });
93
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29waWxvdC1jaGF0LXZpZXctc2Nyb2xsLXRvLWJvdHRvbS1idXR0b24uY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vc3JjL2NvbXBvbmVudHMvY2hhdC9jb3BpbG90LWNoYXQtdmlldy1zY3JvbGwtdG8tYm90dG9tLWJ1dHRvbi5jb21wb25lbnQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUNMLFNBQVMsRUFDVCxLQUFLLEVBQ0wsTUFBTSxFQUNOLFlBQVksRUFDWix1QkFBdUIsRUFDdkIsaUJBQWlCLEVBQ2xCLE1BQU0sZUFBZSxDQUFDO0FBQ3ZCLE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQUMvQyxPQUFPLEVBQUUsbUJBQW1CLEVBQUUsV0FBVyxFQUFFLE1BQU0sZ0JBQWdCLENBQUM7QUFDbEUsT0FBTyxFQUFFLEVBQUUsRUFBRSxNQUFNLGlCQUFpQixDQUFDOzs7QUFFckM7OztHQUdHO0FBb0JILE1BQU0sT0FBTyw0Q0FBNEM7SUFDOUMsVUFBVSxDQUFVO0lBQ3BCLFFBQVEsR0FBWSxLQUFLLENBQUM7SUFDbkMsd0RBQXdEO0lBQy9DLE9BQU8sQ0FBYztJQUU5QixtQ0FBbUM7SUFDekIsT0FBTyxHQUFHLElBQUksWUFBWSxFQUFRLENBQUM7SUFFN0MsaUJBQWlCO0lBQ0UsV0FBVyxHQUFHLFdBQVcsQ0FBQztJQUU3Qyx3Q0FBd0M7SUFDeEMsSUFBSSxhQUFhO1FBQ2YsT0FBTyxFQUFFO1FBQ1AscUJBQXFCO1FBQ3JCLDRCQUE0QjtRQUM1QixvQkFBb0I7UUFDcEIsMkJBQTJCO1FBQzNCLG9CQUFvQjtRQUNwQix1REFBdUQ7UUFDdkQsZUFBZTtRQUNmLHlDQUF5QztRQUN6QyxTQUFTO1FBQ1QsaURBQWlEO1FBQ2pELGFBQWE7UUFDYixtQkFBbUI7UUFDbkIsZUFBZTtRQUNmLHFFQUFxRTtRQUNyRSxpQkFBaUI7UUFDakIsSUFBSSxDQUFDLFVBQVUsQ0FDaEIsQ0FBQztJQUNKLENBQUM7SUFFRCxXQUFXO1FBQ1QsSUFBSSxDQUFDLElBQUksQ0FBQyxRQUFRLEVBQUUsQ0FBQztZQUNuQiw4Q0FBOEM7WUFDOUMsSUFBSSxJQUFJLENBQUMsT0FBTyxFQUFFLENBQUM7Z0JBQ2pCLElBQUksQ0FBQyxPQUFPLEVBQUUsQ0FBQztZQUNqQixDQUFDO1lBQ0QsSUFBSSxDQUFDLE9BQU8sQ0FBQyxJQUFJLEVBQUUsQ0FBQztRQUN0QixDQUFDO0lBQ0gsQ0FBQzt3R0ExQ1UsNENBQTRDOzRGQUE1Qyw0Q0FBNEMsc05BYjdDOzs7Ozs7Ozs7OztHQVdULDJEQWRTLFlBQVksOEJBQUUsbUJBQW1COzs0RkFnQmhDLDRDQUE0QztrQkFuQnhELFNBQVM7bUJBQUM7b0JBQ1QsUUFBUSxFQUFFLDJDQUEyQztvQkFDckQsVUFBVSxFQUFFLElBQUk7b0JBQ2hCLE9BQU8sRUFBRSxDQUFDLFlBQVksRUFBRSxtQkFBbUIsQ0FBQztvQkFDNUMsZUFBZSxFQUFFLHVCQUF1QixDQUFDLE1BQU07b0JBQy9DLGFBQWEsRUFBRSxpQkFBaUIsQ0FBQyxJQUFJO29CQUNyQyxRQUFRLEVBQUU7Ozs7Ozs7Ozs7O0dBV1Q7aUJBQ0Y7OEJBRVUsVUFBVTtzQkFBbEIsS0FBSztnQkFDRyxRQUFRO3NCQUFoQixLQUFLO2dCQUVHLE9BQU87c0JBQWYsS0FBSztnQkFHSSxPQUFPO3NCQUFoQixNQUFNIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHtcbiAgQ29tcG9uZW50LFxuICBJbnB1dCxcbiAgT3V0cHV0LFxuICBFdmVudEVtaXR0ZXIsXG4gIENoYW5nZURldGVjdGlvblN0cmF0ZWd5LFxuICBWaWV3RW5jYXBzdWxhdGlvblxufSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IENvbW1vbk1vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvbW1vbic7XG5pbXBvcnQgeyBMdWNpZGVBbmd1bGFyTW9kdWxlLCBDaGV2cm9uRG93biB9IGZyb20gJ2x1Y2lkZS1hbmd1bGFyJztcbmltcG9ydCB7IGNuIH0gZnJvbSAnLi4vLi4vbGliL3V0aWxzJztcblxuLyoqXG4gKiBTY3JvbGxUb0JvdHRvbUJ1dHRvbiBjb21wb25lbnQgZm9yIENvcGlsb3RDaGF0Vmlld1xuICogTWF0Y2hlcyBSZWFjdCBpbXBsZW1lbnRhdGlvbiBleGFjdGx5IHdpdGggc2FtZSBUYWlsd2luZCBjbGFzc2VzXG4gKi9cbkBDb21wb25lbnQoe1xuICBzZWxlY3RvcjogJ2NvcGlsb3QtY2hhdC12aWV3LXNjcm9sbC10by1ib3R0b20tYnV0dG9uJyxcbiAgc3RhbmRhbG9uZTogdHJ1ZSxcbiAgaW1wb3J0czogW0NvbW1vbk1vZHVsZSwgTHVjaWRlQW5ndWxhck1vZHVsZV0sXG4gIGNoYW5nZURldGVjdGlvbjogQ2hhbmdlRGV0ZWN0aW9uU3RyYXRlZ3kuT25QdXNoLFxuICBlbmNhcHN1bGF0aW9uOiBWaWV3RW5jYXBzdWxhdGlvbi5Ob25lLFxuICB0ZW1wbGF0ZTogYFxuICAgIDxidXR0b25cbiAgICAgIHR5cGU9XCJidXR0b25cIlxuICAgICAgW2NsYXNzXT1cImNvbXB1dGVkQ2xhc3NcIlxuICAgICAgW2Rpc2FibGVkXT1cImRpc2FibGVkXCJcbiAgICAgIChjbGljayk9XCJoYW5kbGVDbGljaygpXCI+XG4gICAgICA8bHVjaWRlLWFuZ3VsYXJcbiAgICAgICAgW2ltZ109XCJDaGV2cm9uRG93blwiXG4gICAgICAgIGNsYXNzPVwidy00IGgtNCB0ZXh0LWdyYXktNjAwIGRhcms6dGV4dC13aGl0ZVwiPlxuICAgICAgPC9sdWNpZGUtYW5ndWxhcj5cbiAgICA8L2J1dHRvbj5cbiAgYFxufSlcbmV4cG9ydCBjbGFzcyBDb3BpbG90Q2hhdFZpZXdTY3JvbGxUb0JvdHRvbUJ1dHRvbkNvbXBvbmVudCB7XG4gIEBJbnB1dCgpIGlucHV0Q2xhc3M/OiBzdHJpbmc7XG4gIEBJbnB1dCgpIGRpc2FibGVkOiBib29sZWFuID0gZmFsc2U7XG4gIC8vIFN1cHBvcnQgZnVuY3Rpb24tc3R5bGUgY2xpY2sgaGFuZGxlciB2aWEgc2xvdCBjb250ZXh0XG4gIEBJbnB1dCgpIG9uQ2xpY2s/OiAoKSA9PiB2b2lkO1xuICBcbiAgLy8gU2ltcGxlLCBpZGlvbWF0aWMgQW5ndWxhciBvdXRwdXRcbiAgQE91dHB1dCgpIGNsaWNrZWQgPSBuZXcgRXZlbnRFbWl0dGVyPHZvaWQ+KCk7XG4gIFxuICAvLyBJY29uIHJlZmVyZW5jZVxuICBwcm90ZWN0ZWQgcmVhZG9ubHkgQ2hldnJvbkRvd24gPSBDaGV2cm9uRG93bjtcbiAgXG4gIC8vIENvbXB1dGVkIGNsYXNzIG1hdGNoaW5nIFJlYWN0IGV4YWN0bHlcbiAgZ2V0IGNvbXB1dGVkQ2xhc3MoKTogc3RyaW5nIHtcbiAgICByZXR1cm4gY24oXG4gICAgICAvLyBCYXNlIGJ1dHRvbiBzdHlsZXNcbiAgICAgICdyb3VuZGVkLWZ1bGwgdy0xMCBoLTEwIHAtMCcsXG4gICAgICAvLyBCYWNrZ3JvdW5kIGNvbG9yc1xuICAgICAgJ2JnLXdoaXRlIGRhcms6YmctZ3JheS05MDAnLFxuICAgICAgLy8gQm9yZGVyIGFuZCBzaGFkb3dcbiAgICAgICdzaGFkb3ctbGcgYm9yZGVyIGJvcmRlci1ncmF5LTIwMCBkYXJrOmJvcmRlci1ncmF5LTcwMCcsXG4gICAgICAvLyBIb3ZlciBzdGF0ZXNcbiAgICAgICdob3ZlcjpiZy1ncmF5LTUwIGRhcms6aG92ZXI6YmctZ3JheS04MDAnLFxuICAgICAgLy8gTGF5b3V0XG4gICAgICAnZmxleCBpdGVtcy1jZW50ZXIganVzdGlmeS1jZW50ZXIgY3Vyc29yLXBvaW50ZXInLFxuICAgICAgLy8gVHJhbnNpdGlvblxuICAgICAgJ3RyYW5zaXRpb24tY29sb3JzJyxcbiAgICAgIC8vIEZvY3VzIHN0YXRlc1xuICAgICAgJ2ZvY3VzOm91dGxpbmUtbm9uZSBmb2N1cy12aXNpYmxlOnJpbmctMiBmb2N1cy12aXNpYmxlOnJpbmctb2Zmc2V0LTInLFxuICAgICAgLy8gQ3VzdG9tIGNsYXNzZXNcbiAgICAgIHRoaXMuaW5wdXRDbGFzc1xuICAgICk7XG4gIH1cbiAgXG4gIGhhbmRsZUNsaWNrKCk6IHZvaWQge1xuICAgIGlmICghdGhpcy5kaXNhYmxlZCkge1xuICAgICAgLy8gQ2FsbCBpbnB1dCBoYW5kbGVyIGlmIHByb3ZpZGVkIChzbG90LXN0eWxlKVxuICAgICAgaWYgKHRoaXMub25DbGljaykge1xuICAgICAgICB0aGlzLm9uQ2xpY2soKTtcbiAgICAgIH1cbiAgICAgIHRoaXMuY2xpY2tlZC5lbWl0KCk7XG4gICAgfVxuICB9XG59XG4iXX0=
@@ -0,0 +1,443 @@
1
+ import { Component, Input, Output, EventEmitter, ViewChild, ElementRef, ChangeDetectionStrategy, ViewEncapsulation, signal, computed, inject, PLATFORM_ID, ChangeDetectorRef } from '@angular/core';
2
+ import { CommonModule, isPlatformBrowser } from '@angular/common';
3
+ import { ScrollingModule } from '@angular/cdk/scrolling';
4
+ import { CopilotSlotComponent } from '../../lib/slots/copilot-slot.component';
5
+ import { CopilotChatMessageViewComponent } from './copilot-chat-message-view.component';
6
+ import { CopilotChatViewScrollToBottomButtonComponent } from './copilot-chat-view-scroll-to-bottom-button.component';
7
+ import { StickToBottomDirective } from '../../directives/stick-to-bottom.directive';
8
+ import { ScrollPositionService } from '../../services/scroll-position.service';
9
+ import { cn } from '../../lib/utils';
10
+ import { Subject } from 'rxjs';
11
+ import { takeUntil } from 'rxjs/operators';
12
+ import * as i0 from "@angular/core";
13
+ import * as i1 from "@angular/cdk/scrolling";
14
+ /**
15
+ * ScrollView component for CopilotChatView
16
+ * Handles auto-scrolling and scroll position management
17
+ */
18
+ export class CopilotChatViewScrollViewComponent {
19
+ cdr = inject(ChangeDetectorRef);
20
+ autoScroll = true;
21
+ _inputContainerHeight = 0;
22
+ set inputContainerHeight(value) {
23
+ this._inputContainerHeight = value;
24
+ this.inputContainerHeightSignal.set(value);
25
+ this.cdr.markForCheck();
26
+ }
27
+ get inputContainerHeight() {
28
+ return this._inputContainerHeight;
29
+ }
30
+ isResizing = false;
31
+ inputClass;
32
+ messages = [];
33
+ messageView;
34
+ messageViewClass;
35
+ showCursor = false;
36
+ // Handler availability flags removed in favor of DI service
37
+ // Slot inputs
38
+ scrollToBottomButton;
39
+ scrollToBottomButtonClass;
40
+ // Output events (bubbled from message view)
41
+ assistantMessageThumbsUp = new EventEmitter();
42
+ assistantMessageThumbsDown = new EventEmitter();
43
+ assistantMessageReadAloud = new EventEmitter();
44
+ assistantMessageRegenerate = new EventEmitter();
45
+ userMessageCopy = new EventEmitter();
46
+ userMessageEdit = new EventEmitter();
47
+ // ViewChild references
48
+ scrollContainer;
49
+ contentContainer;
50
+ stickToBottomDirective;
51
+ // Default components
52
+ defaultMessageViewComponent = CopilotChatMessageViewComponent;
53
+ defaultScrollToBottomButtonComponent = CopilotChatViewScrollToBottomButtonComponent;
54
+ // Signals
55
+ hasMounted = signal(false);
56
+ showScrollButton = signal(false);
57
+ isAtBottom = signal(true);
58
+ inputContainerHeightSignal = signal(0);
59
+ paddingBottom = computed(() => this.inputContainerHeightSignal() + 32);
60
+ // Computed class
61
+ computedClass = computed(() => cn(this.inputClass));
62
+ destroy$ = new Subject();
63
+ platformId = inject(PLATFORM_ID);
64
+ scrollPositionService = inject(ScrollPositionService);
65
+ ngOnInit() {
66
+ // Check if we're in the browser
67
+ if (isPlatformBrowser(this.platformId)) {
68
+ // Set mounted after a tick to allow for hydration
69
+ setTimeout(() => {
70
+ this.hasMounted.set(true);
71
+ }, 0);
72
+ }
73
+ }
74
+ ngOnChanges() {
75
+ // Update signals when inputs change
76
+ // Force change detection when inputContainerHeight changes
77
+ if (this.inputContainerHeight !== undefined) {
78
+ this.cdr.detectChanges();
79
+ }
80
+ }
81
+ ngAfterViewInit() {
82
+ if (!this.autoScroll) {
83
+ // Wait for the view to be fully rendered after hasMounted is set
84
+ setTimeout(() => {
85
+ if (this.scrollContainer) {
86
+ // Check initial scroll position
87
+ const initialState = this.scrollPositionService.getScrollState(this.scrollContainer.nativeElement, 10);
88
+ this.showScrollButton.set(!initialState.isAtBottom);
89
+ // Monitor scroll position for manual mode
90
+ this.scrollPositionService.monitorScrollPosition(this.scrollContainer, 10)
91
+ .pipe(takeUntil(this.destroy$))
92
+ .subscribe(state => {
93
+ this.showScrollButton.set(!state.isAtBottom);
94
+ });
95
+ }
96
+ }, 100);
97
+ }
98
+ }
99
+ /**
100
+ * Handle isAtBottom change from StickToBottom directive
101
+ */
102
+ onIsAtBottomChange(isAtBottom) {
103
+ this.isAtBottom.set(isAtBottom);
104
+ }
105
+ /**
106
+ * Scroll to bottom for manual mode
107
+ */
108
+ scrollToBottom() {
109
+ if (this.scrollContainer) {
110
+ this.scrollPositionService.scrollToBottom(this.scrollContainer, true);
111
+ }
112
+ }
113
+ /**
114
+ * Scroll to bottom for stick-to-bottom mode
115
+ */
116
+ scrollToBottomFromStick() {
117
+ if (this.stickToBottomDirective) {
118
+ this.stickToBottomDirective.scrollToBottom('smooth');
119
+ }
120
+ }
121
+ ngOnDestroy() {
122
+ this.destroy$.next();
123
+ this.destroy$.complete();
124
+ }
125
+ // Output maps for slots
126
+ scrollToBottomOutputs = { clicked: () => this.scrollToBottom() };
127
+ scrollToBottomFromStickOutputs = { clicked: () => this.scrollToBottomFromStick() };
128
+ // Context methods for templates
129
+ messageViewContext() {
130
+ return { messages: this.messages, inputClass: this.messageViewClass, showCursor: this.showCursor };
131
+ }
132
+ scrollToBottomContext() {
133
+ return {
134
+ inputClass: this.scrollToBottomButtonClass,
135
+ onClick: () => this.scrollToBottom()
136
+ };
137
+ }
138
+ scrollToBottomFromStickContext() {
139
+ return {
140
+ inputClass: this.scrollToBottomButtonClass,
141
+ onClick: () => this.scrollToBottomFromStick()
142
+ };
143
+ }
144
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CopilotChatViewScrollViewComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
145
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: CopilotChatViewScrollViewComponent, isStandalone: true, selector: "copilot-chat-view-scroll-view", inputs: { autoScroll: "autoScroll", inputContainerHeight: "inputContainerHeight", isResizing: "isResizing", inputClass: "inputClass", messages: "messages", messageView: "messageView", messageViewClass: "messageViewClass", showCursor: "showCursor", scrollToBottomButton: "scrollToBottomButton", scrollToBottomButtonClass: "scrollToBottomButtonClass" }, outputs: { assistantMessageThumbsUp: "assistantMessageThumbsUp", assistantMessageThumbsDown: "assistantMessageThumbsDown", assistantMessageReadAloud: "assistantMessageReadAloud", assistantMessageRegenerate: "assistantMessageRegenerate", userMessageCopy: "userMessageCopy", userMessageEdit: "userMessageEdit" }, providers: [ScrollPositionService], viewQueries: [{ propertyName: "scrollContainer", first: true, predicate: ["scrollContainer"], descendants: true, read: ElementRef }, { propertyName: "contentContainer", first: true, predicate: ["contentContainer"], descendants: true, read: ElementRef }, { propertyName: "stickToBottomDirective", first: true, predicate: StickToBottomDirective, descendants: true }], usesOnChanges: true, ngImport: i0, template: `
146
+ @if (!hasMounted()) {
147
+ <!-- SSR/Initial render without stick-to-bottom -->
148
+ <div class="h-full max-h-full flex flex-col min-h-0 overflow-y-scroll overflow-x-hidden">
149
+ <div class="px-4 sm:px-0">
150
+ <ng-content></ng-content>
151
+ </div>
152
+ </div>
153
+ } @else if (!autoScroll) {
154
+ <!-- Manual scroll mode -->
155
+ <div class="h-full max-h-full flex flex-col min-h-0 relative">
156
+ <div
157
+ #scrollContainer
158
+ cdkScrollable
159
+ [class]="computedClass()"
160
+ class="overflow-y-scroll overflow-x-hidden">
161
+ <div #contentContainer class="px-4 sm:px-0">
162
+ <!-- Content with padding-bottom matching React -->
163
+ <div [style.padding-bottom.px]="paddingBottom()">
164
+ <div class="max-w-3xl mx-auto">
165
+ @if (messageView) {
166
+ <copilot-slot
167
+ [slot]="messageView"
168
+ [context]="messageViewContext()"
169
+ [defaultComponent]="defaultMessageViewComponent">
170
+ </copilot-slot>
171
+ } @else {
172
+ <copilot-chat-message-view
173
+ [messages]="messages"
174
+ [inputClass]="messageViewClass"
175
+ [showCursor]="showCursor"
176
+ (assistantMessageThumbsUp)="assistantMessageThumbsUp.emit($event)"
177
+ (assistantMessageThumbsDown)="assistantMessageThumbsDown.emit($event)"
178
+ (assistantMessageReadAloud)="assistantMessageReadAloud.emit($event)"
179
+ (assistantMessageRegenerate)="assistantMessageRegenerate.emit($event)"
180
+ (userMessageCopy)="userMessageCopy.emit($event)"
181
+ (userMessageEdit)="userMessageEdit.emit($event)">
182
+ </copilot-chat-message-view>
183
+ }
184
+ </div>
185
+ </div>
186
+ </div>
187
+ </div>
188
+
189
+ <!-- Scroll to bottom button for manual mode, OUTSIDE scrollable content -->
190
+ @if (showScrollButton() && !isResizing) {
191
+ <div
192
+ class="absolute inset-x-0 flex justify-center z-30"
193
+ [style.bottom.px]="inputContainerHeightSignal() + 16">
194
+ <copilot-slot
195
+ [slot]="scrollToBottomButton"
196
+ [context]="scrollToBottomContext()"
197
+ [defaultComponent]="defaultScrollToBottomButtonComponent"
198
+ [outputs]="scrollToBottomOutputs">
199
+ </copilot-slot>
200
+ </div>
201
+ }
202
+ </div>
203
+ } @else {
204
+ <!-- Auto-scroll mode with StickToBottom directive -->
205
+ <div class="h-full max-h-full flex flex-col min-h-0 relative">
206
+ <div
207
+ #scrollContainer
208
+ cdkScrollable
209
+ copilotStickToBottom
210
+ [enabled]="autoScroll"
211
+ [threshold]="10"
212
+ [debounceMs]="0"
213
+ [initialBehavior]="'smooth'"
214
+ [resizeBehavior]="'smooth'"
215
+ (isAtBottomChange)="onIsAtBottomChange($event)"
216
+ [class]="computedClass()"
217
+ class="overflow-y-scroll overflow-x-hidden">
218
+
219
+ <!-- Scrollable content wrapper -->
220
+ <div class="px-4 sm:px-0">
221
+ <!-- Content with padding-bottom matching React -->
222
+ <div [style.padding-bottom.px]="paddingBottom()">
223
+ <div class="max-w-3xl mx-auto">
224
+ @if (messageView) {
225
+ <copilot-slot
226
+ [slot]="messageView"
227
+ [context]="messageViewContext()"
228
+ [defaultComponent]="defaultMessageViewComponent">
229
+ </copilot-slot>
230
+ } @else {
231
+ <copilot-chat-message-view
232
+ [messages]="messages"
233
+ [inputClass]="messageViewClass"
234
+ [showCursor]="showCursor"
235
+ (assistantMessageThumbsUp)="assistantMessageThumbsUp.emit($event)"
236
+ (assistantMessageThumbsDown)="assistantMessageThumbsDown.emit($event)"
237
+ (assistantMessageReadAloud)="assistantMessageReadAloud.emit($event)"
238
+ (assistantMessageRegenerate)="assistantMessageRegenerate.emit($event)"
239
+ (userMessageCopy)="userMessageCopy.emit($event)"
240
+ (userMessageEdit)="userMessageEdit.emit($event)">
241
+ </copilot-chat-message-view>
242
+ }
243
+ </div>
244
+ </div>
245
+ </div>
246
+ </div>
247
+
248
+ <!-- Scroll to bottom button - hidden during resize, OUTSIDE scrollable content -->
249
+ @if (!isAtBottom() && !isResizing) {
250
+ <div
251
+ class="absolute inset-x-0 flex justify-center z-30"
252
+ [style.bottom.px]="inputContainerHeightSignal() + 16">
253
+ <copilot-slot
254
+ [slot]="scrollToBottomButton"
255
+ [context]="scrollToBottomFromStickContext()"
256
+ [defaultComponent]="defaultScrollToBottomButtonComponent"
257
+ [outputs]="scrollToBottomFromStickOutputs">
258
+ </copilot-slot>
259
+ </div>
260
+ }
261
+ </div>
262
+ }
263
+ `, isInline: true, dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "ngmodule", type: ScrollingModule }, { kind: "directive", type: i1.CdkScrollable, selector: "[cdk-scrollable], [cdkScrollable]" }, { kind: "component", type: CopilotSlotComponent, selector: "copilot-slot", inputs: ["slot", "context", "defaultComponent", "outputs"] }, { kind: "component", type: CopilotChatMessageViewComponent, selector: "copilot-chat-message-view", inputs: ["messages", "showCursor", "isLoading", "inputClass", "assistantMessageComponent", "assistantMessageTemplate", "assistantMessageClass", "userMessageComponent", "userMessageTemplate", "userMessageClass", "cursorComponent", "cursorTemplate", "cursorClass"], outputs: ["assistantMessageThumbsUp", "assistantMessageThumbsDown", "assistantMessageReadAloud", "assistantMessageRegenerate", "userMessageCopy", "userMessageEdit"] }, { kind: "directive", type: StickToBottomDirective, selector: "[copilotStickToBottom]", inputs: ["enabled", "threshold", "initialBehavior", "resizeBehavior", "debounceMs"], outputs: ["isAtBottomChange", "scrollToBottomRequested"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None });
264
+ }
265
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CopilotChatViewScrollViewComponent, decorators: [{
266
+ type: Component,
267
+ args: [{
268
+ selector: 'copilot-chat-view-scroll-view',
269
+ standalone: true,
270
+ imports: [
271
+ CommonModule,
272
+ ScrollingModule,
273
+ CopilotSlotComponent,
274
+ CopilotChatMessageViewComponent,
275
+ CopilotChatViewScrollToBottomButtonComponent,
276
+ StickToBottomDirective
277
+ ],
278
+ changeDetection: ChangeDetectionStrategy.OnPush,
279
+ encapsulation: ViewEncapsulation.None,
280
+ providers: [ScrollPositionService],
281
+ template: `
282
+ @if (!hasMounted()) {
283
+ <!-- SSR/Initial render without stick-to-bottom -->
284
+ <div class="h-full max-h-full flex flex-col min-h-0 overflow-y-scroll overflow-x-hidden">
285
+ <div class="px-4 sm:px-0">
286
+ <ng-content></ng-content>
287
+ </div>
288
+ </div>
289
+ } @else if (!autoScroll) {
290
+ <!-- Manual scroll mode -->
291
+ <div class="h-full max-h-full flex flex-col min-h-0 relative">
292
+ <div
293
+ #scrollContainer
294
+ cdkScrollable
295
+ [class]="computedClass()"
296
+ class="overflow-y-scroll overflow-x-hidden">
297
+ <div #contentContainer class="px-4 sm:px-0">
298
+ <!-- Content with padding-bottom matching React -->
299
+ <div [style.padding-bottom.px]="paddingBottom()">
300
+ <div class="max-w-3xl mx-auto">
301
+ @if (messageView) {
302
+ <copilot-slot
303
+ [slot]="messageView"
304
+ [context]="messageViewContext()"
305
+ [defaultComponent]="defaultMessageViewComponent">
306
+ </copilot-slot>
307
+ } @else {
308
+ <copilot-chat-message-view
309
+ [messages]="messages"
310
+ [inputClass]="messageViewClass"
311
+ [showCursor]="showCursor"
312
+ (assistantMessageThumbsUp)="assistantMessageThumbsUp.emit($event)"
313
+ (assistantMessageThumbsDown)="assistantMessageThumbsDown.emit($event)"
314
+ (assistantMessageReadAloud)="assistantMessageReadAloud.emit($event)"
315
+ (assistantMessageRegenerate)="assistantMessageRegenerate.emit($event)"
316
+ (userMessageCopy)="userMessageCopy.emit($event)"
317
+ (userMessageEdit)="userMessageEdit.emit($event)">
318
+ </copilot-chat-message-view>
319
+ }
320
+ </div>
321
+ </div>
322
+ </div>
323
+ </div>
324
+
325
+ <!-- Scroll to bottom button for manual mode, OUTSIDE scrollable content -->
326
+ @if (showScrollButton() && !isResizing) {
327
+ <div
328
+ class="absolute inset-x-0 flex justify-center z-30"
329
+ [style.bottom.px]="inputContainerHeightSignal() + 16">
330
+ <copilot-slot
331
+ [slot]="scrollToBottomButton"
332
+ [context]="scrollToBottomContext()"
333
+ [defaultComponent]="defaultScrollToBottomButtonComponent"
334
+ [outputs]="scrollToBottomOutputs">
335
+ </copilot-slot>
336
+ </div>
337
+ }
338
+ </div>
339
+ } @else {
340
+ <!-- Auto-scroll mode with StickToBottom directive -->
341
+ <div class="h-full max-h-full flex flex-col min-h-0 relative">
342
+ <div
343
+ #scrollContainer
344
+ cdkScrollable
345
+ copilotStickToBottom
346
+ [enabled]="autoScroll"
347
+ [threshold]="10"
348
+ [debounceMs]="0"
349
+ [initialBehavior]="'smooth'"
350
+ [resizeBehavior]="'smooth'"
351
+ (isAtBottomChange)="onIsAtBottomChange($event)"
352
+ [class]="computedClass()"
353
+ class="overflow-y-scroll overflow-x-hidden">
354
+
355
+ <!-- Scrollable content wrapper -->
356
+ <div class="px-4 sm:px-0">
357
+ <!-- Content with padding-bottom matching React -->
358
+ <div [style.padding-bottom.px]="paddingBottom()">
359
+ <div class="max-w-3xl mx-auto">
360
+ @if (messageView) {
361
+ <copilot-slot
362
+ [slot]="messageView"
363
+ [context]="messageViewContext()"
364
+ [defaultComponent]="defaultMessageViewComponent">
365
+ </copilot-slot>
366
+ } @else {
367
+ <copilot-chat-message-view
368
+ [messages]="messages"
369
+ [inputClass]="messageViewClass"
370
+ [showCursor]="showCursor"
371
+ (assistantMessageThumbsUp)="assistantMessageThumbsUp.emit($event)"
372
+ (assistantMessageThumbsDown)="assistantMessageThumbsDown.emit($event)"
373
+ (assistantMessageReadAloud)="assistantMessageReadAloud.emit($event)"
374
+ (assistantMessageRegenerate)="assistantMessageRegenerate.emit($event)"
375
+ (userMessageCopy)="userMessageCopy.emit($event)"
376
+ (userMessageEdit)="userMessageEdit.emit($event)">
377
+ </copilot-chat-message-view>
378
+ }
379
+ </div>
380
+ </div>
381
+ </div>
382
+ </div>
383
+
384
+ <!-- Scroll to bottom button - hidden during resize, OUTSIDE scrollable content -->
385
+ @if (!isAtBottom() && !isResizing) {
386
+ <div
387
+ class="absolute inset-x-0 flex justify-center z-30"
388
+ [style.bottom.px]="inputContainerHeightSignal() + 16">
389
+ <copilot-slot
390
+ [slot]="scrollToBottomButton"
391
+ [context]="scrollToBottomFromStickContext()"
392
+ [defaultComponent]="defaultScrollToBottomButtonComponent"
393
+ [outputs]="scrollToBottomFromStickOutputs">
394
+ </copilot-slot>
395
+ </div>
396
+ }
397
+ </div>
398
+ }
399
+ `
400
+ }]
401
+ }], propDecorators: { autoScroll: [{
402
+ type: Input
403
+ }], inputContainerHeight: [{
404
+ type: Input
405
+ }], isResizing: [{
406
+ type: Input
407
+ }], inputClass: [{
408
+ type: Input
409
+ }], messages: [{
410
+ type: Input
411
+ }], messageView: [{
412
+ type: Input
413
+ }], messageViewClass: [{
414
+ type: Input
415
+ }], showCursor: [{
416
+ type: Input
417
+ }], scrollToBottomButton: [{
418
+ type: Input
419
+ }], scrollToBottomButtonClass: [{
420
+ type: Input
421
+ }], assistantMessageThumbsUp: [{
422
+ type: Output
423
+ }], assistantMessageThumbsDown: [{
424
+ type: Output
425
+ }], assistantMessageReadAloud: [{
426
+ type: Output
427
+ }], assistantMessageRegenerate: [{
428
+ type: Output
429
+ }], userMessageCopy: [{
430
+ type: Output
431
+ }], userMessageEdit: [{
432
+ type: Output
433
+ }], scrollContainer: [{
434
+ type: ViewChild,
435
+ args: ['scrollContainer', { read: ElementRef }]
436
+ }], contentContainer: [{
437
+ type: ViewChild,
438
+ args: ['contentContainer', { read: ElementRef }]
439
+ }], stickToBottomDirective: [{
440
+ type: ViewChild,
441
+ args: [StickToBottomDirective]
442
+ }] } });
443
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"copilot-chat-view-scroll-view.component.js","sourceRoot":"","sources":["../../../../src/components/chat/copilot-chat-view-scroll-view.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  Input,\n  Output,\n  EventEmitter,\n  ViewChild,\n  ElementRef,\n  ChangeDetectionStrategy,\n  ViewEncapsulation,\n  signal,\n  computed,\n  OnInit,\n  OnChanges,\n  AfterViewInit,\n  OnDestroy,\n  inject,\n  PLATFORM_ID,\n  ChangeDetectorRef\n} from '@angular/core';\nimport { CommonModule, isPlatformBrowser } from '@angular/common';\nimport { CdkScrollable, ScrollingModule } from '@angular/cdk/scrolling';\nimport { CopilotSlotComponent } from '../../lib/slots/copilot-slot.component';\nimport { CopilotChatMessageViewComponent } from './copilot-chat-message-view.component';\nimport { CopilotChatViewScrollToBottomButtonComponent } from './copilot-chat-view-scroll-to-bottom-button.component';\nimport { StickToBottomDirective } from '../../directives/stick-to-bottom.directive';\nimport { ScrollPositionService } from '../../services/scroll-position.service';\nimport { Message } from '@ag-ui/client';\nimport { cn } from '../../lib/utils';\nimport { Subject } from 'rxjs';\nimport { takeUntil } from 'rxjs/operators';\n\n/**\n * ScrollView component for CopilotChatView\n * Handles auto-scrolling and scroll position management\n */\n@Component({\n  selector: 'copilot-chat-view-scroll-view',\n  standalone: true,\n  imports: [\n    CommonModule,\n    ScrollingModule,\n    CopilotSlotComponent,\n    CopilotChatMessageViewComponent,\n    CopilotChatViewScrollToBottomButtonComponent,\n    StickToBottomDirective\n  ],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  encapsulation: ViewEncapsulation.None,\n  providers: [ScrollPositionService],\n  template: `\n    @if (!hasMounted()) {\n      <!-- SSR/Initial render without stick-to-bottom -->\n      <div class=\"h-full max-h-full flex flex-col min-h-0 overflow-y-scroll overflow-x-hidden\">\n        <div class=\"px-4 sm:px-0\">\n          <ng-content></ng-content>\n        </div>\n      </div>\n    } @else if (!autoScroll) {\n      <!-- Manual scroll mode -->\n      <div class=\"h-full max-h-full flex flex-col min-h-0 relative\">\n        <div \n          #scrollContainer\n          cdkScrollable\n          [class]=\"computedClass()\"\n          class=\"overflow-y-scroll overflow-x-hidden\">\n          <div #contentContainer class=\"px-4 sm:px-0\">\n            <!-- Content with padding-bottom matching React -->\n            <div [style.padding-bottom.px]=\"paddingBottom()\">\n              <div class=\"max-w-3xl mx-auto\">\n                @if (messageView) {\n                  <copilot-slot\n                    [slot]=\"messageView\"\n                    [context]=\"messageViewContext()\"\n                    [defaultComponent]=\"defaultMessageViewComponent\">\n                  </copilot-slot>\n                } @else {\n                  <copilot-chat-message-view\n                    [messages]=\"messages\"\n                    [inputClass]=\"messageViewClass\"\n                    [showCursor]=\"showCursor\"\n                    (assistantMessageThumbsUp)=\"assistantMessageThumbsUp.emit($event)\"\n                    (assistantMessageThumbsDown)=\"assistantMessageThumbsDown.emit($event)\"\n                    (assistantMessageReadAloud)=\"assistantMessageReadAloud.emit($event)\"\n                    (assistantMessageRegenerate)=\"assistantMessageRegenerate.emit($event)\"\n                    (userMessageCopy)=\"userMessageCopy.emit($event)\"\n                    (userMessageEdit)=\"userMessageEdit.emit($event)\">\n                  </copilot-chat-message-view>\n                }\n              </div>\n            </div>\n          </div>\n        </div>\n        \n        <!-- Scroll to bottom button for manual mode, OUTSIDE scrollable content -->\n        @if (showScrollButton() && !isResizing) {\n          <div\n            class=\"absolute inset-x-0 flex justify-center z-30\"\n            [style.bottom.px]=\"inputContainerHeightSignal() + 16\">\n            <copilot-slot\n              [slot]=\"scrollToBottomButton\"\n              [context]=\"scrollToBottomContext()\"\n              [defaultComponent]=\"defaultScrollToBottomButtonComponent\"\n              [outputs]=\"scrollToBottomOutputs\">\n            </copilot-slot>\n          </div>\n        }\n      </div>\n    } @else {\n      <!-- Auto-scroll mode with StickToBottom directive -->\n      <div class=\"h-full max-h-full flex flex-col min-h-0 relative\">\n        <div \n          #scrollContainer\n          cdkScrollable\n          copilotStickToBottom\n          [enabled]=\"autoScroll\"\n          [threshold]=\"10\"\n          [debounceMs]=\"0\"\n          [initialBehavior]=\"'smooth'\"\n          [resizeBehavior]=\"'smooth'\"\n          (isAtBottomChange)=\"onIsAtBottomChange($event)\"\n          [class]=\"computedClass()\"\n          class=\"overflow-y-scroll overflow-x-hidden\">\n          \n          <!-- Scrollable content wrapper -->\n          <div class=\"px-4 sm:px-0\">\n            <!-- Content with padding-bottom matching React -->\n            <div [style.padding-bottom.px]=\"paddingBottom()\">\n              <div class=\"max-w-3xl mx-auto\">\n                @if (messageView) {\n                  <copilot-slot\n                    [slot]=\"messageView\"\n                    [context]=\"messageViewContext()\"\n                    [defaultComponent]=\"defaultMessageViewComponent\">\n                  </copilot-slot>\n                } @else {\n                  <copilot-chat-message-view\n                    [messages]=\"messages\"\n                    [inputClass]=\"messageViewClass\"\n                    [showCursor]=\"showCursor\"\n                    (assistantMessageThumbsUp)=\"assistantMessageThumbsUp.emit($event)\"\n                    (assistantMessageThumbsDown)=\"assistantMessageThumbsDown.emit($event)\"\n                    (assistantMessageReadAloud)=\"assistantMessageReadAloud.emit($event)\"\n                    (assistantMessageRegenerate)=\"assistantMessageRegenerate.emit($event)\"\n                    (userMessageCopy)=\"userMessageCopy.emit($event)\"\n                    (userMessageEdit)=\"userMessageEdit.emit($event)\">\n                  </copilot-chat-message-view>\n                }\n              </div>\n            </div>\n          </div>\n        </div>\n        \n        <!-- Scroll to bottom button - hidden during resize, OUTSIDE scrollable content -->\n        @if (!isAtBottom() && !isResizing) {\n          <div\n            class=\"absolute inset-x-0 flex justify-center z-30\"\n            [style.bottom.px]=\"inputContainerHeightSignal() + 16\">\n            <copilot-slot\n              [slot]=\"scrollToBottomButton\"\n              [context]=\"scrollToBottomFromStickContext()\"\n              [defaultComponent]=\"defaultScrollToBottomButtonComponent\"\n              [outputs]=\"scrollToBottomFromStickOutputs\">\n            </copilot-slot>\n          </div>\n        }\n      </div>\n    }\n  `\n})\nexport class CopilotChatViewScrollViewComponent implements OnInit, OnChanges, AfterViewInit, OnDestroy {\n  private cdr = inject(ChangeDetectorRef);\n  \n  @Input() autoScroll: boolean = true;\n  \n  private _inputContainerHeight: number = 0;\n  @Input() \n  set inputContainerHeight(value: number) {\n    this._inputContainerHeight = value;\n    this.inputContainerHeightSignal.set(value);\n    this.cdr.markForCheck();\n  }\n  get inputContainerHeight(): number {\n    return this._inputContainerHeight;\n  }\n  \n  @Input() isResizing: boolean = false;\n  @Input() inputClass?: string;\n  @Input() messages: Message[] = [];\n  @Input() messageView?: any;\n  @Input() messageViewClass?: string;\n  @Input() showCursor: boolean = false;\n  \n  // Handler availability flags removed in favor of DI service\n  \n  // Slot inputs\n  @Input() scrollToBottomButton?: any;\n  @Input() scrollToBottomButtonClass?: string;\n  \n  // Output events (bubbled from message view)\n  @Output() assistantMessageThumbsUp = new EventEmitter<{ message: Message }>();\n  @Output() assistantMessageThumbsDown = new EventEmitter<{ message: Message }>();\n  @Output() assistantMessageReadAloud = new EventEmitter<{ message: Message }>();\n  @Output() assistantMessageRegenerate = new EventEmitter<{ message: Message }>();\n  @Output() userMessageCopy = new EventEmitter<{ message: Message }>();\n  @Output() userMessageEdit = new EventEmitter<{ message: Message }>();\n  \n  // ViewChild references\n  @ViewChild('scrollContainer', { read: ElementRef }) scrollContainer?: ElementRef<HTMLElement>;\n  @ViewChild('contentContainer', { read: ElementRef }) contentContainer?: ElementRef<HTMLElement>;\n  @ViewChild(StickToBottomDirective) stickToBottomDirective?: StickToBottomDirective;\n  \n  // Default components\n  protected readonly defaultMessageViewComponent = CopilotChatMessageViewComponent;\n  protected readonly defaultScrollToBottomButtonComponent = CopilotChatViewScrollToBottomButtonComponent;\n  \n  // Signals\n  protected hasMounted = signal(false);\n  protected showScrollButton = signal(false);\n  protected isAtBottom = signal(true);\n  protected inputContainerHeightSignal = signal(0);\n  protected paddingBottom = computed(() => this.inputContainerHeightSignal() + 32);\n  \n  // Computed class\n  protected computedClass = computed(() => \n    cn(this.inputClass)\n  );\n  \n  private destroy$ = new Subject<void>();\n  private platformId = inject(PLATFORM_ID);\n  private scrollPositionService = inject(ScrollPositionService);\n  \n  ngOnInit(): void {\n    // Check if we're in the browser\n    if (isPlatformBrowser(this.platformId)) {\n      // Set mounted after a tick to allow for hydration\n      setTimeout(() => {\n        this.hasMounted.set(true);\n      }, 0);\n    }\n  }\n  \n  ngOnChanges(): void {\n    // Update signals when inputs change\n    // Force change detection when inputContainerHeight changes\n    if (this.inputContainerHeight !== undefined) {\n      this.cdr.detectChanges();\n    }\n  }\n  \n  ngAfterViewInit(): void {\n    if (!this.autoScroll) {\n      // Wait for the view to be fully rendered after hasMounted is set\n      setTimeout(() => {\n        if (this.scrollContainer) {\n          // Check initial scroll position\n          const initialState = this.scrollPositionService.getScrollState(this.scrollContainer.nativeElement, 10);\n          this.showScrollButton.set(!initialState.isAtBottom);\n          \n          // Monitor scroll position for manual mode\n          this.scrollPositionService.monitorScrollPosition(this.scrollContainer, 10)\n            .pipe(takeUntil(this.destroy$))\n            .subscribe(state => {\n              this.showScrollButton.set(!state.isAtBottom);\n            });\n        }\n      }, 100);\n    }\n  }\n  \n  /**\n   * Handle isAtBottom change from StickToBottom directive\n   */\n  onIsAtBottomChange(isAtBottom: boolean): void {\n    this.isAtBottom.set(isAtBottom);\n  }\n  \n  /**\n   * Scroll to bottom for manual mode\n   */\n  scrollToBottom(): void {\n    if (this.scrollContainer) {\n      this.scrollPositionService.scrollToBottom(this.scrollContainer, true);\n    }\n  }\n  \n  /**\n   * Scroll to bottom for stick-to-bottom mode\n   */\n  scrollToBottomFromStick(): void {\n    if (this.stickToBottomDirective) {\n      this.stickToBottomDirective.scrollToBottom('smooth');\n    }\n  }\n  \n  ngOnDestroy(): void {\n    this.destroy$.next();\n    this.destroy$.complete();\n  }\n  \n  // Output maps for slots\n  scrollToBottomOutputs = { clicked: () => this.scrollToBottom() };\n  scrollToBottomFromStickOutputs = { clicked: () => this.scrollToBottomFromStick() };\n  \n  // Context methods for templates\n  messageViewContext(): any {\n    return { messages: this.messages, inputClass: this.messageViewClass, showCursor: this.showCursor };\n  }\n  \n  scrollToBottomContext(): any {\n    return { \n      inputClass: this.scrollToBottomButtonClass,\n      onClick: () => this.scrollToBottom()\n    };\n  }\n  \n  scrollToBottomFromStickContext(): any {\n    return { \n      inputClass: this.scrollToBottomButtonClass,\n      onClick: () => this.scrollToBottomFromStick()\n    };\n  }\n}\n"]}