@copilotkitnext/angular 0.0.9-alpha.3 → 0.0.11

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 (210) hide show
  1. package/dist/esm2022/index.mjs +2 -70
  2. package/dist/esm2022/lib/agent.mjs +73 -0
  3. package/dist/esm2022/lib/chat-config.mjs +35 -0
  4. package/dist/esm2022/lib/chat-state.mjs +18 -0
  5. package/dist/esm2022/lib/components/chat/copilot-chat-assistant-message-buttons.mjs +344 -0
  6. package/dist/esm2022/lib/components/chat/copilot-chat-assistant-message-renderer.mjs +260 -0
  7. package/dist/esm2022/lib/components/chat/copilot-chat-assistant-message-toolbar.mjs +22 -0
  8. package/dist/esm2022/{components/chat/copilot-chat-assistant-message.component.mjs → lib/components/chat/copilot-chat-assistant-message.mjs} +216 -240
  9. package/dist/esm2022/lib/components/chat/copilot-chat-assistant-message.types.mjs +2 -0
  10. package/dist/esm2022/lib/components/chat/copilot-chat-audio-recorder.mjs +196 -0
  11. package/dist/esm2022/lib/components/chat/copilot-chat-buttons.mjs +299 -0
  12. package/dist/esm2022/lib/components/chat/copilot-chat-input-defaults.mjs +39 -0
  13. package/dist/esm2022/lib/components/chat/copilot-chat-input.mjs +634 -0
  14. package/dist/esm2022/lib/components/chat/copilot-chat-input.types.mjs +10 -0
  15. package/dist/esm2022/lib/components/chat/copilot-chat-message-view-cursor.mjs +27 -0
  16. package/dist/esm2022/lib/components/chat/copilot-chat-message-view.mjs +268 -0
  17. package/dist/esm2022/lib/components/chat/copilot-chat-message-view.types.mjs +2 -0
  18. package/dist/esm2022/lib/components/chat/copilot-chat-textarea.mjs +139 -0
  19. package/dist/esm2022/lib/components/chat/copilot-chat-tool-calls-view.mjs +36 -0
  20. package/dist/esm2022/lib/components/chat/copilot-chat-toolbar.mjs +20 -0
  21. package/dist/esm2022/lib/components/chat/copilot-chat-tools-menu.mjs +203 -0
  22. package/dist/esm2022/lib/components/chat/copilot-chat-user-message-branch-navigation.mjs +118 -0
  23. package/dist/esm2022/lib/components/chat/copilot-chat-user-message-buttons.mjs +182 -0
  24. package/dist/esm2022/lib/components/chat/copilot-chat-user-message-renderer.mjs +28 -0
  25. package/dist/esm2022/lib/components/chat/copilot-chat-user-message-toolbar.mjs +25 -0
  26. package/dist/esm2022/lib/components/chat/copilot-chat-user-message.mjs +306 -0
  27. package/dist/esm2022/lib/components/chat/copilot-chat-user-message.types.mjs +2 -0
  28. package/dist/esm2022/lib/components/chat/copilot-chat-view-disclaimer.mjs +48 -0
  29. package/dist/esm2022/lib/components/chat/copilot-chat-view-feather.mjs +41 -0
  30. package/dist/esm2022/lib/components/chat/copilot-chat-view-handlers.mjs +19 -0
  31. package/dist/esm2022/lib/components/chat/copilot-chat-view-input-container.mjs +96 -0
  32. package/dist/esm2022/lib/components/chat/copilot-chat-view-scroll-to-bottom-button.mjs +89 -0
  33. package/dist/esm2022/lib/components/chat/copilot-chat-view-scroll-view.mjs +456 -0
  34. package/dist/esm2022/lib/components/chat/copilot-chat-view.mjs +404 -0
  35. package/dist/esm2022/lib/components/chat/copilot-chat-view.types.mjs +2 -0
  36. package/dist/esm2022/lib/components/chat/copilot-chat.mjs +167 -0
  37. package/dist/esm2022/lib/config.mjs +9 -0
  38. package/dist/esm2022/lib/copilotkit.mjs +124 -0
  39. package/dist/esm2022/lib/directives/copilotkit-agent-context.mjs +130 -0
  40. package/dist/esm2022/lib/directives/stick-to-bottom.mjs +170 -0
  41. package/dist/esm2022/lib/directives/tooltip.mjs +217 -0
  42. package/dist/esm2022/lib/human-in-the-loop.mjs +19 -0
  43. package/dist/esm2022/lib/render-tool-calls.mjs +131 -0
  44. package/dist/esm2022/lib/resize-observer.mjs +152 -0
  45. package/dist/esm2022/lib/scroll-position.mjs +124 -0
  46. package/dist/esm2022/lib/slots/copilot-slot.mjs +156 -0
  47. package/dist/esm2022/lib/slots/index.mjs +4 -0
  48. package/dist/esm2022/lib/slots/slot.types.mjs +3 -3
  49. package/dist/esm2022/lib/slots/slot.utils.mjs +19 -15
  50. package/dist/esm2022/lib/tools.mjs +31 -0
  51. package/dist/esm2022/lib/utils.mjs +3 -3
  52. package/dist/esm2022/public-api.mjs +47 -0
  53. package/dist/fesm2022/copilotkitnext-angular.mjs +5249 -8271
  54. package/dist/fesm2022/copilotkitnext-angular.mjs.map +1 -1
  55. package/dist/index.d.ts +1 -55
  56. package/dist/lib/agent.d.ts +53 -0
  57. package/dist/{core/chat-configuration/chat-configuration.types.d.ts → lib/chat-config.d.ts} +4 -8
  58. package/dist/lib/chat-state.d.ts +10 -0
  59. package/dist/lib/components/chat/copilot-chat-assistant-message-buttons.d.ts +68 -0
  60. package/dist/lib/components/chat/copilot-chat-assistant-message-renderer.d.ts +26 -0
  61. package/dist/lib/components/chat/copilot-chat-assistant-message-toolbar.d.ts +7 -0
  62. package/dist/lib/components/chat/copilot-chat-assistant-message.d.ts +178 -0
  63. package/dist/{components → lib/components}/chat/copilot-chat-assistant-message.types.d.ts +1 -1
  64. package/dist/{components/chat/copilot-chat-audio-recorder.component.d.ts → lib/components/chat/copilot-chat-audio-recorder.d.ts} +10 -10
  65. package/dist/lib/components/chat/copilot-chat-buttons.d.ts +65 -0
  66. package/dist/lib/components/chat/copilot-chat-input-defaults.d.ts +38 -0
  67. package/dist/lib/components/chat/copilot-chat-input.d.ts +133 -0
  68. package/dist/{components → lib/components}/chat/copilot-chat-input.types.d.ts +11 -11
  69. package/dist/lib/components/chat/copilot-chat-message-view-cursor.d.ts +11 -0
  70. package/dist/{components/chat/copilot-chat-message-view.component.d.ts → lib/components/chat/copilot-chat-message-view.d.ts} +68 -36
  71. package/dist/{components → lib/components}/chat/copilot-chat-message-view.types.d.ts +2 -2
  72. package/dist/lib/components/chat/copilot-chat-textarea.d.ts +41 -0
  73. package/dist/lib/components/chat/copilot-chat-tool-calls-view.d.ts +55 -0
  74. package/dist/lib/components/chat/copilot-chat-toolbar.d.ts +7 -0
  75. package/dist/lib/components/chat/copilot-chat-tools-menu.d.ts +20 -0
  76. package/dist/lib/components/chat/copilot-chat-user-message-branch-navigation.d.ts +20 -0
  77. package/dist/lib/components/chat/copilot-chat-user-message-buttons.d.ts +35 -0
  78. package/dist/lib/components/chat/copilot-chat-user-message-renderer.d.ts +8 -0
  79. package/dist/lib/components/chat/copilot-chat-user-message-toolbar.d.ts +7 -0
  80. package/dist/lib/components/chat/copilot-chat-user-message.d.ts +55 -0
  81. package/dist/{components → lib/components}/chat/copilot-chat-user-message.types.d.ts +2 -2
  82. package/dist/lib/components/chat/copilot-chat-view-disclaimer.d.ts +15 -0
  83. package/dist/{components/chat/copilot-chat-view-feather.component.d.ts → lib/components/chat/copilot-chat-view-feather.d.ts} +6 -6
  84. package/dist/{components/chat/copilot-chat-view-handlers.service.d.ts → lib/components/chat/copilot-chat-view-handlers.d.ts} +3 -3
  85. package/dist/lib/components/chat/copilot-chat-view-input-container.d.ts +23 -0
  86. package/dist/lib/components/chat/copilot-chat-view-scroll-to-bottom-button.d.ts +16 -0
  87. package/dist/lib/components/chat/copilot-chat-view-scroll-view.d.ts +114 -0
  88. package/dist/lib/components/chat/copilot-chat-view.d.ts +239 -0
  89. package/dist/{components → lib/components}/chat/copilot-chat-view.types.d.ts +2 -2
  90. package/dist/lib/components/chat/copilot-chat.d.ts +67 -0
  91. package/dist/lib/config.d.ts +16 -0
  92. package/dist/lib/copilotkit.d.ts +29 -0
  93. package/dist/{directives/copilotkit-agent-context.directive.d.ts → lib/directives/copilotkit-agent-context.d.ts} +5 -5
  94. package/dist/lib/directives/stick-to-bottom.d.ts +62 -0
  95. package/dist/lib/directives/tooltip.d.ts +33 -0
  96. package/dist/lib/human-in-the-loop.d.ts +13 -0
  97. package/dist/lib/render-tool-calls.d.ts +75 -0
  98. package/dist/{services/resize-observer.service.d.ts → lib/resize-observer.d.ts} +2 -2
  99. package/dist/{services/scroll-position.service.d.ts → lib/scroll-position.d.ts} +6 -6
  100. package/dist/lib/slots/copilot-slot.d.ts +34 -0
  101. package/dist/lib/slots/index.d.ts +3 -0
  102. package/dist/lib/slots/slot.types.d.ts +1 -1
  103. package/dist/lib/slots/slot.utils.d.ts +6 -4
  104. package/dist/lib/tools.d.ts +63 -0
  105. package/dist/lib/utils.d.ts +1 -1
  106. package/dist/public-api.d.ts +46 -0
  107. package/dist/styles.css +0 -69
  108. package/package.json +5 -5
  109. package/dist/components/chat/copilot-chat-assistant-message-buttons.component.d.ts +0 -75
  110. package/dist/components/chat/copilot-chat-assistant-message-renderer.component.d.ts +0 -31
  111. package/dist/components/chat/copilot-chat-assistant-message-toolbar.component.d.ts +0 -8
  112. package/dist/components/chat/copilot-chat-assistant-message.component.d.ts +0 -132
  113. package/dist/components/chat/copilot-chat-buttons.component.d.ts +0 -66
  114. package/dist/components/chat/copilot-chat-input-defaults.d.ts +0 -37
  115. package/dist/components/chat/copilot-chat-input.component.d.ts +0 -133
  116. package/dist/components/chat/copilot-chat-message-view-cursor.component.d.ts +0 -15
  117. package/dist/components/chat/copilot-chat-textarea.component.d.ts +0 -45
  118. package/dist/components/chat/copilot-chat-tool-calls-view.component.d.ts +0 -35
  119. package/dist/components/chat/copilot-chat-toolbar.component.d.ts +0 -8
  120. package/dist/components/chat/copilot-chat-tools-menu.component.d.ts +0 -20
  121. package/dist/components/chat/copilot-chat-user-message-branch-navigation.component.d.ts +0 -23
  122. package/dist/components/chat/copilot-chat-user-message-buttons.component.d.ts +0 -39
  123. package/dist/components/chat/copilot-chat-user-message-renderer.component.d.ts +0 -9
  124. package/dist/components/chat/copilot-chat-user-message-toolbar.component.d.ts +0 -8
  125. package/dist/components/chat/copilot-chat-user-message.component.d.ts +0 -55
  126. package/dist/components/chat/copilot-chat-view-disclaimer.component.d.ts +0 -15
  127. package/dist/components/chat/copilot-chat-view-input-container.component.d.ts +0 -23
  128. package/dist/components/chat/copilot-chat-view-scroll-to-bottom-button.component.d.ts +0 -17
  129. package/dist/components/chat/copilot-chat-view-scroll-view.component.d.ts +0 -84
  130. package/dist/components/chat/copilot-chat-view.component.d.ts +0 -205
  131. package/dist/components/chat/copilot-chat.component.d.ts +0 -36
  132. package/dist/components/copilotkit-tool-render.component.d.ts +0 -25
  133. package/dist/core/chat-configuration/chat-configuration.providers.d.ts +0 -54
  134. package/dist/core/chat-configuration/chat-configuration.service.d.ts +0 -75
  135. package/dist/core/copilotkit.providers.d.ts +0 -13
  136. package/dist/core/copilotkit.service.d.ts +0 -119
  137. package/dist/core/copilotkit.types.d.ts +0 -81
  138. package/dist/directives/copilotkit-agent.directive.d.ts +0 -106
  139. package/dist/directives/copilotkit-chat-config.directive.d.ts +0 -84
  140. package/dist/directives/copilotkit-config.directive.d.ts +0 -44
  141. package/dist/directives/copilotkit-frontend-tool.directive.d.ts +0 -25
  142. package/dist/directives/copilotkit-human-in-the-loop.directive.d.ts +0 -124
  143. package/dist/directives/stick-to-bottom.directive.d.ts +0 -62
  144. package/dist/esm2022/components/chat/copilot-chat-assistant-message-buttons.component.mjs +0 -384
  145. package/dist/esm2022/components/chat/copilot-chat-assistant-message-renderer.component.mjs +0 -286
  146. package/dist/esm2022/components/chat/copilot-chat-assistant-message-toolbar.component.mjs +0 -27
  147. package/dist/esm2022/components/chat/copilot-chat-assistant-message.types.mjs +0 -2
  148. package/dist/esm2022/components/chat/copilot-chat-audio-recorder.component.mjs +0 -202
  149. package/dist/esm2022/components/chat/copilot-chat-buttons.component.mjs +0 -321
  150. package/dist/esm2022/components/chat/copilot-chat-input-defaults.mjs +0 -38
  151. package/dist/esm2022/components/chat/copilot-chat-input.component.mjs +0 -666
  152. package/dist/esm2022/components/chat/copilot-chat-input.types.mjs +0 -10
  153. package/dist/esm2022/components/chat/copilot-chat-message-view-cursor.component.mjs +0 -45
  154. package/dist/esm2022/components/chat/copilot-chat-message-view.component.mjs +0 -296
  155. package/dist/esm2022/components/chat/copilot-chat-message-view.types.mjs +0 -2
  156. package/dist/esm2022/components/chat/copilot-chat-textarea.component.mjs +0 -188
  157. package/dist/esm2022/components/chat/copilot-chat-tool-calls-view.component.mjs +0 -222
  158. package/dist/esm2022/components/chat/copilot-chat-toolbar.component.mjs +0 -25
  159. package/dist/esm2022/components/chat/copilot-chat-tools-menu.component.mjs +0 -199
  160. package/dist/esm2022/components/chat/copilot-chat-user-message-branch-navigation.component.mjs +0 -137
  161. package/dist/esm2022/components/chat/copilot-chat-user-message-buttons.component.mjs +0 -207
  162. package/dist/esm2022/components/chat/copilot-chat-user-message-renderer.component.mjs +0 -35
  163. package/dist/esm2022/components/chat/copilot-chat-user-message-toolbar.component.mjs +0 -34
  164. package/dist/esm2022/components/chat/copilot-chat-user-message.component.mjs +0 -341
  165. package/dist/esm2022/components/chat/copilot-chat-user-message.types.mjs +0 -2
  166. package/dist/esm2022/components/chat/copilot-chat-view-disclaimer.component.mjs +0 -52
  167. package/dist/esm2022/components/chat/copilot-chat-view-feather.component.mjs +0 -55
  168. package/dist/esm2022/components/chat/copilot-chat-view-handlers.service.mjs +0 -19
  169. package/dist/esm2022/components/chat/copilot-chat-view-input-container.component.mjs +0 -110
  170. package/dist/esm2022/components/chat/copilot-chat-view-scroll-to-bottom-button.component.mjs +0 -93
  171. package/dist/esm2022/components/chat/copilot-chat-view-scroll-view.component.mjs +0 -443
  172. package/dist/esm2022/components/chat/copilot-chat-view.component.mjs +0 -479
  173. package/dist/esm2022/components/chat/copilot-chat-view.types.mjs +0 -2
  174. package/dist/esm2022/components/chat/copilot-chat.component.mjs +0 -220
  175. package/dist/esm2022/components/copilotkit-tool-render.component.mjs +0 -150
  176. package/dist/esm2022/core/chat-configuration/chat-configuration.providers.mjs +0 -65
  177. package/dist/esm2022/core/chat-configuration/chat-configuration.service.mjs +0 -145
  178. package/dist/esm2022/core/chat-configuration/chat-configuration.types.mjs +0 -26
  179. package/dist/esm2022/core/copilotkit.providers.mjs +0 -34
  180. package/dist/esm2022/core/copilotkit.service.mjs +0 -411
  181. package/dist/esm2022/core/copilotkit.types.mjs +0 -13
  182. package/dist/esm2022/directives/copilotkit-agent-context.directive.mjs +0 -130
  183. package/dist/esm2022/directives/copilotkit-agent.directive.mjs +0 -221
  184. package/dist/esm2022/directives/copilotkit-chat-config.directive.mjs +0 -218
  185. package/dist/esm2022/directives/copilotkit-config.directive.mjs +0 -94
  186. package/dist/esm2022/directives/copilotkit-frontend-tool.directive.mjs +0 -128
  187. package/dist/esm2022/directives/copilotkit-human-in-the-loop.directive.mjs +0 -265
  188. package/dist/esm2022/directives/stick-to-bottom.directive.mjs +0 -181
  189. package/dist/esm2022/lib/directives/tooltip.directive.mjs +0 -211
  190. package/dist/esm2022/lib/slots/copilot-slot.component.mjs +0 -154
  191. package/dist/esm2022/services/resize-observer.service.mjs +0 -152
  192. package/dist/esm2022/services/scroll-position.service.mjs +0 -124
  193. package/dist/esm2022/types/frontend-tool.mjs +0 -2
  194. package/dist/esm2022/types/human-in-the-loop.mjs +0 -2
  195. package/dist/esm2022/utils/agent-context.utils.mjs +0 -114
  196. package/dist/esm2022/utils/agent.utils.mjs +0 -212
  197. package/dist/esm2022/utils/chat-config.utils.mjs +0 -186
  198. package/dist/esm2022/utils/copilotkit.utils.mjs +0 -20
  199. package/dist/esm2022/utils/frontend-tool.utils.mjs +0 -224
  200. package/dist/esm2022/utils/human-in-the-loop.utils.mjs +0 -293
  201. package/dist/lib/directives/tooltip.directive.d.ts +0 -33
  202. package/dist/lib/slots/copilot-slot.component.d.ts +0 -34
  203. package/dist/types/frontend-tool.d.ts +0 -37
  204. package/dist/types/human-in-the-loop.d.ts +0 -44
  205. package/dist/utils/agent-context.utils.d.ts +0 -75
  206. package/dist/utils/agent.utils.d.ts +0 -108
  207. package/dist/utils/chat-config.utils.d.ts +0 -166
  208. package/dist/utils/copilotkit.utils.d.ts +0 -16
  209. package/dist/utils/frontend-tool.utils.d.ts +0 -119
  210. package/dist/utils/human-in-the-loop.utils.d.ts +0 -92
@@ -0,0 +1,152 @@
1
+ import { Injectable, ElementRef } from "@angular/core";
2
+ import { Subject, BehaviorSubject } from "rxjs";
3
+ import { debounceTime, takeUntil, distinctUntilChanged } from "rxjs/operators";
4
+ import * as i0 from "@angular/core";
5
+ export class ResizeObserverService {
6
+ ngZone;
7
+ destroy$ = new Subject();
8
+ observers = new Map();
9
+ resizeStates = new Map();
10
+ resizeTimeouts = new Map();
11
+ constructor(ngZone) {
12
+ this.ngZone = ngZone;
13
+ }
14
+ /**
15
+ * Observe element resize with debouncing and resizing state
16
+ * @param element Element to observe
17
+ * @param debounceMs Debounce time (default 250ms)
18
+ * @param resizingDurationMs How long to show "isResizing" state (default 250ms)
19
+ */
20
+ observeElement(element, debounceMs = 0, resizingDurationMs = 250) {
21
+ const el = element instanceof ElementRef ? element.nativeElement : element;
22
+ // Return existing observer if already observing
23
+ if (this.resizeStates.has(el)) {
24
+ return this.resizeStates.get(el).asObservable();
25
+ }
26
+ // Create new subject for this element
27
+ const resizeState$ = new BehaviorSubject({
28
+ width: el.offsetWidth,
29
+ height: el.offsetHeight,
30
+ isResizing: false,
31
+ });
32
+ this.resizeStates.set(el, resizeState$);
33
+ // Create ResizeObserver
34
+ const resizeObserver = new ResizeObserver((entries) => {
35
+ if (entries.length === 0)
36
+ return;
37
+ const entry = entries[0];
38
+ if (!entry)
39
+ return;
40
+ const { width, height } = entry.contentRect;
41
+ this.ngZone.run(() => {
42
+ // Clear existing timeout
43
+ const existingTimeout = this.resizeTimeouts.get(el);
44
+ if (existingTimeout) {
45
+ clearTimeout(existingTimeout);
46
+ }
47
+ // Update state with isResizing = true
48
+ resizeState$.next({
49
+ width,
50
+ height,
51
+ isResizing: true,
52
+ });
53
+ // Set timeout to clear isResizing flag
54
+ if (resizingDurationMs > 0) {
55
+ const timeout = window.setTimeout(() => {
56
+ resizeState$.next({
57
+ width,
58
+ height,
59
+ isResizing: false,
60
+ });
61
+ this.resizeTimeouts.delete(el);
62
+ }, resizingDurationMs);
63
+ this.resizeTimeouts.set(el, timeout);
64
+ }
65
+ else {
66
+ // If no duration, immediately set isResizing to false
67
+ resizeState$.next({
68
+ width,
69
+ height,
70
+ isResizing: false,
71
+ });
72
+ }
73
+ });
74
+ });
75
+ // Start observing
76
+ resizeObserver.observe(el);
77
+ this.observers.set(el, resizeObserver);
78
+ // Return observable with debouncing if specified
79
+ const observable = resizeState$.asObservable().pipe(debounceMs > 0 ? debounceTime(debounceMs) : (source) => source, distinctUntilChanged((a, b) => a.width === b.width &&
80
+ a.height === b.height &&
81
+ a.isResizing === b.isResizing), takeUntil(this.destroy$));
82
+ return observable;
83
+ }
84
+ /**
85
+ * Stop observing an element
86
+ * @param element Element to stop observing
87
+ */
88
+ unobserve(element) {
89
+ const el = element instanceof ElementRef ? element.nativeElement : element;
90
+ // Clear timeout if exists
91
+ const timeout = this.resizeTimeouts.get(el);
92
+ if (timeout) {
93
+ clearTimeout(timeout);
94
+ this.resizeTimeouts.delete(el);
95
+ }
96
+ // Disconnect observer
97
+ const observer = this.observers.get(el);
98
+ if (observer) {
99
+ observer.disconnect();
100
+ this.observers.delete(el);
101
+ }
102
+ // Complete and remove subject
103
+ const subject = this.resizeStates.get(el);
104
+ if (subject) {
105
+ subject.complete();
106
+ this.resizeStates.delete(el);
107
+ }
108
+ }
109
+ /**
110
+ * Get current size of element
111
+ * @param element Element to measure
112
+ */
113
+ getCurrentSize(element) {
114
+ const el = element instanceof ElementRef ? element.nativeElement : element;
115
+ return {
116
+ width: el.offsetWidth,
117
+ height: el.offsetHeight,
118
+ };
119
+ }
120
+ /**
121
+ * Get current resize state of element
122
+ * @param element Element to check
123
+ */
124
+ getCurrentState(element) {
125
+ const el = element instanceof ElementRef ? element.nativeElement : element;
126
+ const subject = this.resizeStates.get(el);
127
+ return subject ? subject.value : null;
128
+ }
129
+ ngOnDestroy() {
130
+ // Clear all timeouts
131
+ this.resizeTimeouts.forEach((timeout) => clearTimeout(timeout));
132
+ this.resizeTimeouts.clear();
133
+ // Disconnect all observers
134
+ this.observers.forEach((observer) => observer.disconnect());
135
+ this.observers.clear();
136
+ // Complete all subjects
137
+ this.resizeStates.forEach((subject) => subject.complete());
138
+ this.resizeStates.clear();
139
+ // Complete destroy subject
140
+ this.destroy$.next();
141
+ this.destroy$.complete();
142
+ }
143
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ResizeObserverService, deps: [{ token: i0.NgZone }], target: i0.ɵɵFactoryTarget.Injectable });
144
+ static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ResizeObserverService, providedIn: "root" });
145
+ }
146
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ResizeObserverService, decorators: [{
147
+ type: Injectable,
148
+ args: [{
149
+ providedIn: "root",
150
+ }]
151
+ }], ctorParameters: () => [{ type: i0.NgZone }] });
152
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"resize-observer.js","sourceRoot":"","sources":["../../../src/lib/resize-observer.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, ElementRef, NgZone, OnDestroy } from \"@angular/core\";\nimport { Observable, Subject, BehaviorSubject } from \"rxjs\";\nimport { debounceTime, takeUntil, distinctUntilChanged } from \"rxjs/operators\";\n\nexport interface ResizeState {\n  width: number;\n  height: number;\n  isResizing: boolean;\n}\n\n@Injectable({\n  providedIn: \"root\",\n})\nexport class ResizeObserverService implements OnDestroy {\n  private destroy$ = new Subject<void>();\n  private observers = new Map<HTMLElement, ResizeObserver>();\n  private resizeStates = new Map<HTMLElement, BehaviorSubject<ResizeState>>();\n  private resizeTimeouts = new Map<HTMLElement, number>();\n\n  constructor(private ngZone: NgZone) {}\n\n  /**\n   * Observe element resize with debouncing and resizing state\n   * @param element Element to observe\n   * @param debounceMs Debounce time (default 250ms)\n   * @param resizingDurationMs How long to show \"isResizing\" state (default 250ms)\n   */\n  observeElement(\n    element: ElementRef<HTMLElement> | HTMLElement,\n    debounceMs: number = 0,\n    resizingDurationMs: number = 250\n  ): Observable<ResizeState> {\n    const el = element instanceof ElementRef ? element.nativeElement : element;\n\n    // Return existing observer if already observing\n    if (this.resizeStates.has(el)) {\n      return this.resizeStates.get(el)!.asObservable();\n    }\n\n    // Create new subject for this element\n    const resizeState$ = new BehaviorSubject<ResizeState>({\n      width: el.offsetWidth,\n      height: el.offsetHeight,\n      isResizing: false,\n    });\n\n    this.resizeStates.set(el, resizeState$);\n\n    // Create ResizeObserver\n    const resizeObserver = new ResizeObserver((entries) => {\n      if (entries.length === 0) return;\n\n      const entry = entries[0];\n      if (!entry) return;\n\n      const { width, height } = entry.contentRect;\n\n      this.ngZone.run(() => {\n        // Clear existing timeout\n        const existingTimeout = this.resizeTimeouts.get(el);\n        if (existingTimeout) {\n          clearTimeout(existingTimeout);\n        }\n\n        // Update state with isResizing = true\n        resizeState$.next({\n          width,\n          height,\n          isResizing: true,\n        });\n\n        // Set timeout to clear isResizing flag\n        if (resizingDurationMs > 0) {\n          const timeout = window.setTimeout(() => {\n            resizeState$.next({\n              width,\n              height,\n              isResizing: false,\n            });\n            this.resizeTimeouts.delete(el);\n          }, resizingDurationMs);\n\n          this.resizeTimeouts.set(el, timeout);\n        } else {\n          // If no duration, immediately set isResizing to false\n          resizeState$.next({\n            width,\n            height,\n            isResizing: false,\n          });\n        }\n      });\n    });\n\n    // Start observing\n    resizeObserver.observe(el);\n    this.observers.set(el, resizeObserver);\n\n    // Return observable with debouncing if specified\n    const observable = resizeState$.asObservable().pipe(\n      debounceMs > 0 ? debounceTime(debounceMs) : (source) => source,\n      distinctUntilChanged(\n        (a, b) =>\n          a.width === b.width &&\n          a.height === b.height &&\n          a.isResizing === b.isResizing\n      ),\n      takeUntil(this.destroy$)\n    );\n\n    return observable;\n  }\n\n  /**\n   * Stop observing an element\n   * @param element Element to stop observing\n   */\n  unobserve(element: ElementRef<HTMLElement> | HTMLElement): void {\n    const el = element instanceof ElementRef ? element.nativeElement : element;\n\n    // Clear timeout if exists\n    const timeout = this.resizeTimeouts.get(el);\n    if (timeout) {\n      clearTimeout(timeout);\n      this.resizeTimeouts.delete(el);\n    }\n\n    // Disconnect observer\n    const observer = this.observers.get(el);\n    if (observer) {\n      observer.disconnect();\n      this.observers.delete(el);\n    }\n\n    // Complete and remove subject\n    const subject = this.resizeStates.get(el);\n    if (subject) {\n      subject.complete();\n      this.resizeStates.delete(el);\n    }\n  }\n\n  /**\n   * Get current size of element\n   * @param element Element to measure\n   */\n  getCurrentSize(element: ElementRef<HTMLElement> | HTMLElement): {\n    width: number;\n    height: number;\n  } {\n    const el = element instanceof ElementRef ? element.nativeElement : element;\n    return {\n      width: el.offsetWidth,\n      height: el.offsetHeight,\n    };\n  }\n\n  /**\n   * Get current resize state of element\n   * @param element Element to check\n   */\n  getCurrentState(\n    element: ElementRef<HTMLElement> | HTMLElement\n  ): ResizeState | null {\n    const el = element instanceof ElementRef ? element.nativeElement : element;\n    const subject = this.resizeStates.get(el);\n    return subject ? subject.value : null;\n  }\n\n  ngOnDestroy(): void {\n    // Clear all timeouts\n    this.resizeTimeouts.forEach((timeout) => clearTimeout(timeout));\n    this.resizeTimeouts.clear();\n\n    // Disconnect all observers\n    this.observers.forEach((observer) => observer.disconnect());\n    this.observers.clear();\n\n    // Complete all subjects\n    this.resizeStates.forEach((subject) => subject.complete());\n    this.resizeStates.clear();\n\n    // Complete destroy subject\n    this.destroy$.next();\n    this.destroy$.complete();\n  }\n}\n"]}
@@ -0,0 +1,124 @@
1
+ import { Injectable, ElementRef } from "@angular/core";
2
+ import { Subject, BehaviorSubject, fromEvent, merge, animationFrameScheduler, } from "rxjs";
3
+ import { takeUntil, debounceTime, throttleTime, distinctUntilChanged, map, startWith, } from "rxjs/operators";
4
+ import * as i0 from "@angular/core";
5
+ import * as i1 from "@angular/cdk/scrolling";
6
+ export class ScrollPosition {
7
+ scrollDispatcher;
8
+ viewportRuler;
9
+ ngZone;
10
+ destroy$ = new Subject();
11
+ scrollStateSubject = new BehaviorSubject({
12
+ isAtBottom: true,
13
+ scrollTop: 0,
14
+ scrollHeight: 0,
15
+ clientHeight: 0,
16
+ });
17
+ scrollState$ = this.scrollStateSubject.asObservable();
18
+ constructor(scrollDispatcher, viewportRuler, ngZone) {
19
+ this.scrollDispatcher = scrollDispatcher;
20
+ this.viewportRuler = viewportRuler;
21
+ this.ngZone = ngZone;
22
+ }
23
+ /**
24
+ * Monitor scroll position of an element
25
+ * @param element The element to monitor
26
+ * @param threshold Pixels from bottom to consider "at bottom" (default 10)
27
+ */
28
+ monitorScrollPosition(element, threshold = 10) {
29
+ const el = element instanceof ElementRef ? element.nativeElement : element;
30
+ // Create scroll observable
31
+ const scroll$ = merge(fromEvent(el, "scroll"), this.viewportRuler.change(150) // Monitor viewport changes
32
+ ).pipe(startWith(null), // Emit initial state
33
+ throttleTime(16, animationFrameScheduler, { trailing: true }), // ~60fps
34
+ map(() => this.getScrollState(el, threshold)), distinctUntilChanged((a, b) => a.isAtBottom === b.isAtBottom &&
35
+ a.scrollTop === b.scrollTop &&
36
+ a.scrollHeight === b.scrollHeight), takeUntil(this.destroy$));
37
+ // Subscribe and update subject
38
+ scroll$.subscribe((state) => {
39
+ this.scrollStateSubject.next(state);
40
+ });
41
+ return scroll$;
42
+ }
43
+ /**
44
+ * Scroll element to bottom with smooth animation
45
+ * @param element The element to scroll
46
+ * @param smooth Whether to use smooth scrolling
47
+ */
48
+ scrollToBottom(element, smooth = true) {
49
+ const el = element instanceof ElementRef ? element.nativeElement : element;
50
+ this.ngZone.runOutsideAngular(() => {
51
+ if (smooth && "scrollBehavior" in document.documentElement.style) {
52
+ el.scrollTo({
53
+ top: el.scrollHeight,
54
+ behavior: "smooth",
55
+ });
56
+ }
57
+ else {
58
+ el.scrollTop = el.scrollHeight;
59
+ }
60
+ });
61
+ }
62
+ /**
63
+ * Check if element is at bottom
64
+ * @param element The element to check
65
+ * @param threshold Pixels from bottom to consider "at bottom"
66
+ */
67
+ isAtBottom(element, threshold = 10) {
68
+ const el = element instanceof ElementRef ? element.nativeElement : element;
69
+ return this.getScrollState(el, threshold).isAtBottom;
70
+ }
71
+ /**
72
+ * Get current scroll state of element
73
+ */
74
+ getScrollState(element, threshold) {
75
+ const scrollTop = element.scrollTop;
76
+ const scrollHeight = element.scrollHeight;
77
+ const clientHeight = element.clientHeight;
78
+ const distanceFromBottom = scrollHeight - scrollTop - clientHeight;
79
+ const isAtBottom = distanceFromBottom <= threshold;
80
+ return {
81
+ isAtBottom,
82
+ scrollTop,
83
+ scrollHeight,
84
+ clientHeight,
85
+ };
86
+ }
87
+ /**
88
+ * Create a ResizeObserver for element size changes
89
+ * @param element The element to observe
90
+ * @param debounceMs Debounce time in milliseconds
91
+ */
92
+ observeResize(element, debounceMs = 250) {
93
+ const el = element instanceof ElementRef ? element.nativeElement : element;
94
+ const resize$ = new Subject();
95
+ const resizeObserver = new ResizeObserver((entries) => {
96
+ const entry = entries[0];
97
+ if (entry) {
98
+ this.ngZone.run(() => {
99
+ resize$.next(entry);
100
+ });
101
+ }
102
+ });
103
+ resizeObserver.observe(el);
104
+ // Cleanup on destroy
105
+ this.destroy$.subscribe(() => {
106
+ resizeObserver.disconnect();
107
+ });
108
+ return resize$.pipe(debounceTime(debounceMs), takeUntil(this.destroy$));
109
+ }
110
+ ngOnDestroy() {
111
+ this.destroy$.next();
112
+ this.destroy$.complete();
113
+ this.scrollStateSubject.complete();
114
+ }
115
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ScrollPosition, deps: [{ token: i1.ScrollDispatcher }, { token: i1.ViewportRuler }, { token: i0.NgZone }], target: i0.ɵɵFactoryTarget.Injectable });
116
+ static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ScrollPosition, providedIn: "root" });
117
+ }
118
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ScrollPosition, decorators: [{
119
+ type: Injectable,
120
+ args: [{
121
+ providedIn: "root",
122
+ }]
123
+ }], ctorParameters: () => [{ type: i1.ScrollDispatcher }, { type: i1.ViewportRuler }, { type: i0.NgZone }] });
124
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"scroll-position.js","sourceRoot":"","sources":["../../../src/lib/scroll-position.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, ElementRef, NgZone, OnDestroy } from \"@angular/core\";\nimport { ScrollDispatcher, ViewportRuler } from \"@angular/cdk/scrolling\";\nimport {\n  Observable,\n  Subject,\n  BehaviorSubject,\n  fromEvent,\n  merge,\n  animationFrameScheduler,\n} from \"rxjs\";\nimport {\n  takeUntil,\n  debounceTime,\n  throttleTime,\n  distinctUntilChanged,\n  map,\n  startWith,\n} from \"rxjs/operators\";\n\nexport interface ScrollState {\n  isAtBottom: boolean;\n  scrollTop: number;\n  scrollHeight: number;\n  clientHeight: number;\n}\n\n@Injectable({\n  providedIn: \"root\",\n})\nexport class ScrollPosition implements OnDestroy {\n  private destroy$ = new Subject<void>();\n  private scrollStateSubject = new BehaviorSubject<ScrollState>({\n    isAtBottom: true,\n    scrollTop: 0,\n    scrollHeight: 0,\n    clientHeight: 0,\n  });\n\n  public scrollState$ = this.scrollStateSubject.asObservable();\n\n  constructor(\n    private scrollDispatcher: ScrollDispatcher,\n    private viewportRuler: ViewportRuler,\n    private ngZone: NgZone\n  ) {}\n\n  /**\n   * Monitor scroll position of an element\n   * @param element The element to monitor\n   * @param threshold Pixels from bottom to consider \"at bottom\" (default 10)\n   */\n  monitorScrollPosition(\n    element: ElementRef<HTMLElement> | HTMLElement,\n    threshold: number = 10\n  ): Observable<ScrollState> {\n    const el = element instanceof ElementRef ? element.nativeElement : element;\n\n    // Create scroll observable\n    const scroll$ = merge(\n      fromEvent(el, \"scroll\"),\n      this.viewportRuler.change(150) // Monitor viewport changes\n    ).pipe(\n      startWith(null), // Emit initial state\n      throttleTime(16, animationFrameScheduler, { trailing: true }), // ~60fps\n      map(() => this.getScrollState(el, threshold)),\n      distinctUntilChanged(\n        (a, b) =>\n          a.isAtBottom === b.isAtBottom &&\n          a.scrollTop === b.scrollTop &&\n          a.scrollHeight === b.scrollHeight\n      ),\n      takeUntil(this.destroy$)\n    );\n\n    // Subscribe and update subject\n    scroll$.subscribe((state) => {\n      this.scrollStateSubject.next(state);\n    });\n\n    return scroll$;\n  }\n\n  /**\n   * Scroll element to bottom with smooth animation\n   * @param element The element to scroll\n   * @param smooth Whether to use smooth scrolling\n   */\n  scrollToBottom(\n    element: ElementRef<HTMLElement> | HTMLElement,\n    smooth: boolean = true\n  ): void {\n    const el = element instanceof ElementRef ? element.nativeElement : element;\n\n    this.ngZone.runOutsideAngular(() => {\n      if (smooth && \"scrollBehavior\" in document.documentElement.style) {\n        el.scrollTo({\n          top: el.scrollHeight,\n          behavior: \"smooth\",\n        });\n      } else {\n        el.scrollTop = el.scrollHeight;\n      }\n    });\n  }\n\n  /**\n   * Check if element is at bottom\n   * @param element The element to check\n   * @param threshold Pixels from bottom to consider \"at bottom\"\n   */\n  isAtBottom(\n    element: ElementRef<HTMLElement> | HTMLElement,\n    threshold: number = 10\n  ): boolean {\n    const el = element instanceof ElementRef ? element.nativeElement : element;\n    return this.getScrollState(el, threshold).isAtBottom;\n  }\n\n  /**\n   * Get current scroll state of element\n   */\n  public getScrollState(element: HTMLElement, threshold: number): ScrollState {\n    const scrollTop = element.scrollTop;\n    const scrollHeight = element.scrollHeight;\n    const clientHeight = element.clientHeight;\n    const distanceFromBottom = scrollHeight - scrollTop - clientHeight;\n    const isAtBottom = distanceFromBottom <= threshold;\n\n    return {\n      isAtBottom,\n      scrollTop,\n      scrollHeight,\n      clientHeight,\n    };\n  }\n\n  /**\n   * Create a ResizeObserver for element size changes\n   * @param element The element to observe\n   * @param debounceMs Debounce time in milliseconds\n   */\n  observeResize(\n    element: ElementRef<HTMLElement> | HTMLElement,\n    debounceMs: number = 250\n  ): Observable<ResizeObserverEntry> {\n    const el = element instanceof ElementRef ? element.nativeElement : element;\n    const resize$ = new Subject<ResizeObserverEntry>();\n\n    const resizeObserver = new ResizeObserver((entries) => {\n      const entry = entries[0];\n      if (entry) {\n        this.ngZone.run(() => {\n          resize$.next(entry);\n        });\n      }\n    });\n\n    resizeObserver.observe(el);\n\n    // Cleanup on destroy\n    this.destroy$.subscribe(() => {\n      resizeObserver.disconnect();\n    });\n\n    return resize$.pipe(debounceTime(debounceMs), takeUntil(this.destroy$));\n  }\n\n  ngOnDestroy(): void {\n    this.destroy$.next();\n    this.destroy$.complete();\n    this.scrollStateSubject.complete();\n  }\n}\n"]}
@@ -0,0 +1,156 @@
1
+ import { Component, TemplateRef, ViewContainerRef, Inject, ChangeDetectionStrategy, input, ViewChild, } from "@angular/core";
2
+ import { CommonModule } from "@angular/common";
3
+ import { renderSlot } from "./slot.utils";
4
+ import * as i0 from "@angular/core";
5
+ import * as i1 from "@angular/common";
6
+ /**
7
+ * @internal - This component is for internal use only.
8
+ * Simple slot component for rendering custom content or defaults.
9
+ * Supports templates and components only.
10
+ *
11
+ * @example
12
+ * ```html
13
+ * <!-- With template -->
14
+ * <copilot-slot [slot]="sendButtonTemplate" [context]="buttonContext">
15
+ * <button class="default-btn">Default</button>
16
+ * </copilot-slot>
17
+ * ```
18
+ */
19
+ export class CopilotSlot {
20
+ viewContainer;
21
+ cdr;
22
+ slot = input(undefined);
23
+ context = input(undefined);
24
+ defaultComponent = input(undefined);
25
+ outputs = input(undefined);
26
+ slotContainer;
27
+ componentRef;
28
+ constructor(viewContainer, cdr) {
29
+ this.viewContainer = viewContainer;
30
+ this.cdr = cdr;
31
+ }
32
+ ngOnInit() {
33
+ this.renderSlot();
34
+ }
35
+ ngOnChanges(changes) {
36
+ if (changes["slot"]) {
37
+ // Slot changed, need to re-render completely
38
+ this.renderSlot();
39
+ }
40
+ else if (changes["context"] && this.componentRef) {
41
+ // Just context changed, update existing component
42
+ this.updateComponentProps();
43
+ this.cdr.detectChanges();
44
+ }
45
+ else if (changes["context"]) {
46
+ // No component ref yet, render the slot
47
+ this.renderSlot();
48
+ }
49
+ }
50
+ isTemplate(value) {
51
+ return value instanceof TemplateRef;
52
+ }
53
+ renderSlot() {
54
+ // Skip if it's a template (handled by ngTemplateOutlet)
55
+ if (this.slot() && this.isTemplate(this.slot())) {
56
+ this.componentRef = null;
57
+ return;
58
+ }
59
+ // Clear previous content
60
+ this.slotContainer.clear();
61
+ this.componentRef = null;
62
+ // Skip if no slot and no default component
63
+ if (!this.slot() && !this.defaultComponent()) {
64
+ return;
65
+ }
66
+ // Use the utility to render other slot types
67
+ if (this.slot() || this.defaultComponent()) {
68
+ this.componentRef = renderSlot(this.slotContainer, {
69
+ slot: this.slot(),
70
+ defaultComponent: this.defaultComponent(),
71
+ props: this.context(),
72
+ outputs: this.outputs(),
73
+ });
74
+ }
75
+ }
76
+ updateComponentProps() {
77
+ if (!this.componentRef || !this.componentRef.instance) {
78
+ return;
79
+ }
80
+ const props = this.context();
81
+ // Update props using setInput, only for declared inputs
82
+ if (props) {
83
+ const ctor = this.componentRef.instance.constructor;
84
+ const cmpDef = ctor?.ɵcmp;
85
+ const declaredInputs = new Set(Object.keys(cmpDef?.inputs ?? {}));
86
+ if (declaredInputs.has("props")) {
87
+ this.componentRef.setInput("props", props);
88
+ }
89
+ else {
90
+ for (const key in props) {
91
+ if (declaredInputs.has(key)) {
92
+ const value = props[key];
93
+ this.componentRef.setInput(key, value);
94
+ }
95
+ }
96
+ }
97
+ }
98
+ // Trigger change detection
99
+ if (this.componentRef.changeDetectorRef) {
100
+ this.componentRef.changeDetectorRef.detectChanges();
101
+ }
102
+ }
103
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CopilotSlot, deps: [{ token: ViewContainerRef }, { token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
104
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: CopilotSlot, isStandalone: true, selector: "copilot-slot", inputs: { slot: { classPropertyName: "slot", publicName: "slot", isSignal: true, isRequired: false, transformFunction: null }, context: { classPropertyName: "context", publicName: "context", isSignal: true, isRequired: false, transformFunction: null }, defaultComponent: { classPropertyName: "defaultComponent", publicName: "defaultComponent", isSignal: true, isRequired: false, transformFunction: null }, outputs: { classPropertyName: "outputs", publicName: "outputs", isSignal: true, isRequired: false, transformFunction: null } }, viewQueries: [{ propertyName: "slotContainer", first: true, predicate: ["slotContainer"], descendants: true, read: ViewContainerRef, static: true }], usesOnChanges: true, ngImport: i0, template: `
105
+ <!-- If slot template provided, render it -->
106
+ @if (slot() && isTemplate(slot)) {
107
+ <ng-container
108
+ [ngTemplateOutlet]="slot"
109
+ [ngTemplateOutletContext]="context || {}"
110
+ >
111
+ </ng-container>
112
+ }
113
+
114
+ <!-- If not a template, we'll handle in code -->
115
+ <ng-container #slotContainer></ng-container>
116
+
117
+ <!-- Default content (only shown if no slot) -->
118
+ @if (!slot && !defaultComponent) {
119
+ <ng-content></ng-content>
120
+ }
121
+ `, isInline: true, dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
122
+ }
123
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CopilotSlot, decorators: [{
124
+ type: Component,
125
+ args: [{
126
+ standalone: true,
127
+ selector: "copilot-slot",
128
+ imports: [CommonModule],
129
+ template: `
130
+ <!-- If slot template provided, render it -->
131
+ @if (slot() && isTemplate(slot)) {
132
+ <ng-container
133
+ [ngTemplateOutlet]="slot"
134
+ [ngTemplateOutletContext]="context || {}"
135
+ >
136
+ </ng-container>
137
+ }
138
+
139
+ <!-- If not a template, we'll handle in code -->
140
+ <ng-container #slotContainer></ng-container>
141
+
142
+ <!-- Default content (only shown if no slot) -->
143
+ @if (!slot && !defaultComponent) {
144
+ <ng-content></ng-content>
145
+ }
146
+ `,
147
+ changeDetection: ChangeDetectionStrategy.OnPush,
148
+ }]
149
+ }], ctorParameters: () => [{ type: i0.ViewContainerRef, decorators: [{
150
+ type: Inject,
151
+ args: [ViewContainerRef]
152
+ }] }, { type: i0.ChangeDetectorRef }], propDecorators: { slotContainer: [{
153
+ type: ViewChild,
154
+ args: ["slotContainer", { read: ViewContainerRef, static: true }]
155
+ }] } });
156
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"copilot-slot.js","sourceRoot":"","sources":["../../../../src/lib/slots/copilot-slot.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  TemplateRef,\n  ViewContainerRef,\n  OnInit,\n  OnChanges,\n  SimpleChanges,\n  Inject,\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  input,\n  ViewChild,\n} from \"@angular/core\";\nimport { CommonModule } from \"@angular/common\";\nimport { renderSlot } from \"./slot.utils\";\nimport { Type } from \"@angular/core\";\n\n/**\n * @internal - This component is for internal use only.\n * Simple slot component for rendering custom content or defaults.\n * Supports templates and components only.\n *\n * @example\n * ```html\n * <!-- With template -->\n * <copilot-slot [slot]=\"sendButtonTemplate\" [context]=\"buttonContext\">\n *   <button class=\"default-btn\">Default</button>\n * </copilot-slot>\n * ```\n */\n@Component({\n  standalone: true,\n  selector: \"copilot-slot\",\n  imports: [CommonModule],\n  template: `\n    <!-- If slot template provided, render it -->\n    @if (slot() && isTemplate(slot)) {\n      <ng-container\n        [ngTemplateOutlet]=\"slot\"\n        [ngTemplateOutletContext]=\"context || {}\"\n      >\n      </ng-container>\n    }\n\n    <!-- If not a template, we'll handle in code -->\n    <ng-container #slotContainer></ng-container>\n\n    <!-- Default content (only shown if no slot) -->\n    @if (!slot && !defaultComponent) {\n      <ng-content></ng-content>\n    }\n  `,\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class CopilotSlot implements OnInit, OnChanges {\n  slot = input<TemplateRef<any> | Type<any> | undefined>(undefined);\n  context = input<any | undefined>(undefined);\n  defaultComponent = input<Type<any> | undefined>(undefined);\n  outputs = input<Record<string, (event: any) => void> | undefined>(undefined);\n\n  @ViewChild(\"slotContainer\", { read: ViewContainerRef, static: true })\n  private slotContainer!: ViewContainerRef;\n\n  private componentRef?: any;\n\n  constructor(\n    @Inject(ViewContainerRef) private viewContainer: ViewContainerRef,\n    private cdr: ChangeDetectorRef\n  ) {}\n\n  ngOnInit(): void {\n    this.renderSlot();\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes[\"slot\"]) {\n      // Slot changed, need to re-render completely\n      this.renderSlot();\n    } else if (changes[\"context\"] && this.componentRef) {\n      // Just context changed, update existing component\n      this.updateComponentProps();\n      this.cdr.detectChanges();\n    } else if (changes[\"context\"]) {\n      // No component ref yet, render the slot\n      this.renderSlot();\n    }\n  }\n\n  isTemplate(value: any): value is TemplateRef<any> {\n    return value instanceof TemplateRef;\n  }\n\n  private renderSlot(): void {\n    // Skip if it's a template (handled by ngTemplateOutlet)\n    if (this.slot() && this.isTemplate(this.slot())) {\n      this.componentRef = null;\n      return;\n    }\n\n    // Clear previous content\n    this.slotContainer.clear();\n    this.componentRef = null;\n\n    // Skip if no slot and no default component\n    if (!this.slot() && !this.defaultComponent()) {\n      return;\n    }\n\n    // Use the utility to render other slot types\n    if (this.slot() || this.defaultComponent()) {\n      this.componentRef = renderSlot(this.slotContainer, {\n        slot: this.slot(),\n        defaultComponent: this.defaultComponent()!,\n        props: this.context(),\n        outputs: this.outputs(),\n      });\n    }\n  }\n\n  private updateComponentProps(): void {\n    if (!this.componentRef || !this.componentRef.instance) {\n      return;\n    }\n\n    const props = this.context();\n\n    // Update props using setInput, only for declared inputs\n    if (props) {\n      const ctor = this.componentRef.instance.constructor as any;\n      const cmpDef: any = ctor?.ɵcmp;\n      const declaredInputs = new Set<string>(Object.keys(cmpDef?.inputs ?? {}));\n\n      if (declaredInputs.has(\"props\")) {\n        this.componentRef.setInput(\"props\", props);\n      } else {\n        for (const key in props) {\n          if (declaredInputs.has(key)) {\n            const value = props[key];\n            this.componentRef.setInput(key, value);\n          }\n        }\n      }\n    }\n\n    // Trigger change detection\n    if (this.componentRef.changeDetectorRef) {\n      this.componentRef.changeDetectorRef.detectChanges();\n    }\n  }\n}\n"]}
@@ -0,0 +1,4 @@
1
+ export * from "./slot.types";
2
+ export * from "./slot.utils";
3
+ export { CopilotSlot } from "./copilot-slot";
4
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9zcmMvbGliL3Nsb3RzL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLGNBQWMsY0FBYyxDQUFDO0FBQzdCLGNBQWMsY0FBYyxDQUFDO0FBQzdCLE9BQU8sRUFBRSxXQUFXLEVBQUUsTUFBTSxnQkFBZ0IsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCAqIGZyb20gXCIuL3Nsb3QudHlwZXNcIjtcbmV4cG9ydCAqIGZyb20gXCIuL3Nsb3QudXRpbHNcIjtcbmV4cG9ydCB7IENvcGlsb3RTbG90IH0gZnJvbSBcIi4vY29waWxvdC1zbG90XCI7XG4iXX0=
@@ -1,6 +1,6 @@
1
- import { InjectionToken } from '@angular/core';
1
+ import { InjectionToken } from "@angular/core";
2
2
  /**
3
3
  * Injection token for slot configuration
4
4
  */
5
- export const SLOT_CONFIG = new InjectionToken('SLOT_CONFIG');
6
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2xvdC50eXBlcy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3NyYy9saWIvc2xvdHMvc2xvdC50eXBlcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQXFCLGNBQWMsRUFBRSxNQUFNLGVBQWUsQ0FBQztBQWlEbEU7O0dBRUc7QUFDSCxNQUFNLENBQUMsTUFBTSxXQUFXLEdBQUcsSUFBSSxjQUFjLENBQXlDLGFBQWEsQ0FBQyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgVHlwZSwgVGVtcGxhdGVSZWYsIEluamVjdGlvblRva2VuIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5cbi8qKlxuICogUmVwcmVzZW50cyBhIHZhbHVlIHRoYXQgY2FuIGJlIHVzZWQgYXMgYSBzbG90IG92ZXJyaWRlLlxuICogQ2FuIGJlIGEgY29tcG9uZW50IHR5cGUgb3IgdGVtcGxhdGUgcmVmZXJlbmNlIG9ubHkuXG4gKiBAaW50ZXJuYWwgLSBUaGlzIHR5cGUgaXMgZm9yIGludGVybmFsIHVzZSBvbmx5XG4gKi9cbmV4cG9ydCB0eXBlIFNsb3RWYWx1ZTxUID0gYW55PiA9IFxuICB8IFR5cGU8VD5cbiAgfCBUZW1wbGF0ZVJlZjxUPjtcblxuLyoqXG4gKiBDb25maWd1cmF0aW9uIGZvciBhIHNsb3RcbiAqIEBpbnRlcm5hbCAtIFRoaXMgaW50ZXJmYWNlIGlzIGZvciBpbnRlcm5hbCB1c2Ugb25seVxuICovXG5leHBvcnQgaW50ZXJmYWNlIFNsb3RDb25maWc8VCA9IGFueT4ge1xuICB2YWx1ZT86IFNsb3RWYWx1ZTxUPjtcbiAgZGVmYXVsdD86IFR5cGU8VD47XG59XG5cbi8qKlxuICogQ29udGV4dCBwYXNzZWQgdG8gc2xvdCB0ZW1wbGF0ZXNcbiAqL1xuZXhwb3J0IGludGVyZmFjZSBTbG90Q29udGV4dDxUID0gYW55PiB7XG4gICRpbXBsaWNpdDogVDtcbiAgcHJvcHM/OiBQYXJ0aWFsPFQ+O1xuICBba2V5OiBzdHJpbmddOiBhbnk7XG59XG5cbi8qKlxuICogU2xvdCByZWdpc3RyeSBlbnRyeVxuICogQGludGVybmFsIC0gVGhpcyBpbnRlcmZhY2UgaXMgZm9yIGludGVybmFsIHVzZSBvbmx5XG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgU2xvdFJlZ2lzdHJ5RW50cnk8VCA9IGFueT4ge1xuICBjb21wb25lbnQ/OiBUeXBlPFQ+O1xuICB0ZW1wbGF0ZT86IFRlbXBsYXRlUmVmPFQ+O1xufVxuXG4vKipcbiAqIE9wdGlvbnMgZm9yIHJlbmRlcmluZyBhIHNsb3RcbiAqL1xuZXhwb3J0IGludGVyZmFjZSBSZW5kZXJTbG90T3B0aW9uczxUID0gYW55PiB7XG4gIHNsb3Q/OiBTbG90VmFsdWU8VD47XG4gIGRlZmF1bHRDb21wb25lbnQ6IFR5cGU8VD47XG4gIHByb3BzPzogUGFydGlhbDxUPjtcbiAgaW5qZWN0b3I/OiBhbnk7XG4gIG91dHB1dHM/OiBSZWNvcmQ8c3RyaW5nLCAoZXZlbnQ6IGFueSkgPT4gdm9pZD47XG59XG5cbi8qKlxuICogSW5qZWN0aW9uIHRva2VuIGZvciBzbG90IGNvbmZpZ3VyYXRpb25cbiAqL1xuZXhwb3J0IGNvbnN0IFNMT1RfQ09ORklHID0gbmV3IEluamVjdGlvblRva2VuPFJlYWRvbmx5TWFwPHN0cmluZywgU2xvdFJlZ2lzdHJ5RW50cnk+PignU0xPVF9DT05GSUcnKTtcblxuLyoqXG4gKiBUeXBlIGZvciBjb21wb25lbnRzIHdpdGggc2xvdHNcbiAqL1xuZXhwb3J0IHR5cGUgV2l0aFNsb3RzPFMgZXh0ZW5kcyBSZWNvcmQ8c3RyaW5nLCBUeXBlPGFueT4+LCBSZXN0ID0gb2JqZWN0PiA9IHtcbiAgW0sgaW4ga2V5b2YgUyBhcyBgJHtzdHJpbmcgJiBLfUNvbXBvbmVudGBdPzogVHlwZTxhbnk+O1xufSAmIHtcbiAgW0sgaW4ga2V5b2YgUyBhcyBgJHtzdHJpbmcgJiBLfVRlbXBsYXRlYF0/OiBUZW1wbGF0ZVJlZjxhbnk+O1xufSAmIHtcbiAgW0sgaW4ga2V5b2YgUyBhcyBgJHtzdHJpbmcgJiBLfUNsYXNzYF0/OiBzdHJpbmc7XG59ICYgUmVzdDsiXX0=
5
+ export const SLOT_CONFIG = new InjectionToken("SLOT_CONFIG");
6
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2xvdC50eXBlcy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3NyYy9saWIvc2xvdHMvc2xvdC50eXBlcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQXFCLGNBQWMsRUFBRSxNQUFNLGVBQWUsQ0FBQztBQStDbEU7O0dBRUc7QUFDSCxNQUFNLENBQUMsTUFBTSxXQUFXLEdBQUcsSUFBSSxjQUFjLENBRTNDLGFBQWEsQ0FBQyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgVHlwZSwgVGVtcGxhdGVSZWYsIEluamVjdGlvblRva2VuIH0gZnJvbSBcIkBhbmd1bGFyL2NvcmVcIjtcblxuLyoqXG4gKiBSZXByZXNlbnRzIGEgdmFsdWUgdGhhdCBjYW4gYmUgdXNlZCBhcyBhIHNsb3Qgb3ZlcnJpZGUuXG4gKiBDYW4gYmUgYSBjb21wb25lbnQgdHlwZSBvciB0ZW1wbGF0ZSByZWZlcmVuY2Ugb25seS5cbiAqIEBpbnRlcm5hbCAtIFRoaXMgdHlwZSBpcyBmb3IgaW50ZXJuYWwgdXNlIG9ubHlcbiAqL1xuZXhwb3J0IHR5cGUgU2xvdFZhbHVlPFQgPSBhbnk+ID0gVHlwZTxUPiB8IFRlbXBsYXRlUmVmPFQ+O1xuXG4vKipcbiAqIENvbmZpZ3VyYXRpb24gZm9yIGEgc2xvdFxuICogQGludGVybmFsIC0gVGhpcyBpbnRlcmZhY2UgaXMgZm9yIGludGVybmFsIHVzZSBvbmx5XG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgU2xvdENvbmZpZzxUID0gYW55PiB7XG4gIHZhbHVlPzogU2xvdFZhbHVlPFQ+O1xuICBkZWZhdWx0PzogVHlwZTxUPjtcbn1cblxuLyoqXG4gKiBDb250ZXh0IHBhc3NlZCB0byBzbG90IHRlbXBsYXRlc1xuICovXG5leHBvcnQgaW50ZXJmYWNlIFNsb3RDb250ZXh0PFQgPSBhbnk+IHtcbiAgJGltcGxpY2l0OiBUO1xuICBwcm9wcz86IFBhcnRpYWw8VD47XG4gIFtrZXk6IHN0cmluZ106IGFueTtcbn1cblxuLyoqXG4gKiBTbG90IHJlZ2lzdHJ5IGVudHJ5XG4gKiBAaW50ZXJuYWwgLSBUaGlzIGludGVyZmFjZSBpcyBmb3IgaW50ZXJuYWwgdXNlIG9ubHlcbiAqL1xuZXhwb3J0IGludGVyZmFjZSBTbG90UmVnaXN0cnlFbnRyeTxUID0gYW55PiB7XG4gIGNvbXBvbmVudD86IFR5cGU8VD47XG4gIHRlbXBsYXRlPzogVGVtcGxhdGVSZWY8VD47XG59XG5cbi8qKlxuICogT3B0aW9ucyBmb3IgcmVuZGVyaW5nIGEgc2xvdFxuICovXG5leHBvcnQgaW50ZXJmYWNlIFJlbmRlclNsb3RPcHRpb25zPFQgPSBhbnk+IHtcbiAgc2xvdD86IFNsb3RWYWx1ZTxUPjtcbiAgZGVmYXVsdENvbXBvbmVudDogVHlwZTxUPjtcbiAgcHJvcHM/OiBQYXJ0aWFsPFQ+O1xuICBpbmplY3Rvcj86IGFueTtcbiAgb3V0cHV0cz86IFJlY29yZDxzdHJpbmcsIChldmVudDogYW55KSA9PiB2b2lkPjtcbn1cblxuLyoqXG4gKiBJbmplY3Rpb24gdG9rZW4gZm9yIHNsb3QgY29uZmlndXJhdGlvblxuICovXG5leHBvcnQgY29uc3QgU0xPVF9DT05GSUcgPSBuZXcgSW5qZWN0aW9uVG9rZW48XG4gIFJlYWRvbmx5TWFwPHN0cmluZywgU2xvdFJlZ2lzdHJ5RW50cnk+XG4+KFwiU0xPVF9DT05GSUdcIik7XG5cbi8qKlxuICogVHlwZSBmb3IgY29tcG9uZW50cyB3aXRoIHNsb3RzXG4gKi9cbmV4cG9ydCB0eXBlIFdpdGhTbG90czxTIGV4dGVuZHMgUmVjb3JkPHN0cmluZywgVHlwZTxhbnk+PiwgUmVzdCA9IG9iamVjdD4gPSB7XG4gIFtLIGluIGtleW9mIFMgYXMgYCR7c3RyaW5nICYgS31Db21wb25lbnRgXT86IFR5cGU8YW55Pjtcbn0gJiB7XG4gIFtLIGluIGtleW9mIFMgYXMgYCR7c3RyaW5nICYgS31UZW1wbGF0ZWBdPzogVGVtcGxhdGVSZWY8YW55Pjtcbn0gJiB7XG4gIFtLIGluIGtleW9mIFMgYXMgYCR7c3RyaW5nICYgS31DbGFzc2BdPzogc3RyaW5nO1xufSAmIFJlc3Q7XG4iXX0=