@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,144 @@
1
+ import { Component, Input, TemplateRef, ViewContainerRef, Inject, ChangeDetectionStrategy, 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 CopilotSlotComponent {
20
+ viewContainer;
21
+ cdr;
22
+ slot;
23
+ context;
24
+ defaultComponent;
25
+ outputs;
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
82
+ if (props) {
83
+ for (const key in props) {
84
+ const value = props[key];
85
+ this.componentRef.setInput(key, value);
86
+ }
87
+ }
88
+ // Trigger change detection
89
+ if (this.componentRef.changeDetectorRef) {
90
+ this.componentRef.changeDetectorRef.detectChanges();
91
+ }
92
+ }
93
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CopilotSlotComponent, deps: [{ token: ViewContainerRef }, { token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
94
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: CopilotSlotComponent, isStandalone: true, selector: "copilot-slot", inputs: { slot: "slot", context: "context", defaultComponent: "defaultComponent", outputs: "outputs" }, viewQueries: [{ propertyName: "slotContainer", first: true, predicate: ["slotContainer"], descendants: true, read: ViewContainerRef, static: true }], usesOnChanges: true, ngImport: i0, template: `
95
+ <!-- If slot template provided, render it -->
96
+ <ng-container *ngIf="slot && isTemplate(slot)"
97
+ [ngTemplateOutlet]="slot"
98
+ [ngTemplateOutletContext]="context || {}">
99
+ </ng-container>
100
+
101
+ <!-- If not a template, we'll handle in code -->
102
+ <ng-container #slotContainer></ng-container>
103
+
104
+ <!-- Default content (only shown if no slot) -->
105
+ <ng-content *ngIf="!slot && !defaultComponent"></ng-content>
106
+ `, isInline: true, dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
107
+ }
108
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CopilotSlotComponent, decorators: [{
109
+ type: Component,
110
+ args: [{
111
+ selector: 'copilot-slot',
112
+ standalone: true,
113
+ imports: [CommonModule],
114
+ template: `
115
+ <!-- If slot template provided, render it -->
116
+ <ng-container *ngIf="slot && isTemplate(slot)"
117
+ [ngTemplateOutlet]="slot"
118
+ [ngTemplateOutletContext]="context || {}">
119
+ </ng-container>
120
+
121
+ <!-- If not a template, we'll handle in code -->
122
+ <ng-container #slotContainer></ng-container>
123
+
124
+ <!-- Default content (only shown if no slot) -->
125
+ <ng-content *ngIf="!slot && !defaultComponent"></ng-content>
126
+ `,
127
+ changeDetection: ChangeDetectionStrategy.OnPush
128
+ }]
129
+ }], ctorParameters: () => [{ type: i0.ViewContainerRef, decorators: [{
130
+ type: Inject,
131
+ args: [ViewContainerRef]
132
+ }] }, { type: i0.ChangeDetectorRef }], propDecorators: { slot: [{
133
+ type: Input
134
+ }], context: [{
135
+ type: Input
136
+ }], defaultComponent: [{
137
+ type: Input
138
+ }], outputs: [{
139
+ type: Input
140
+ }], slotContainer: [{
141
+ type: ViewChild,
142
+ args: ['slotContainer', { read: ViewContainerRef, static: true }]
143
+ }] } });
144
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"copilot-slot.component.js","sourceRoot":"","sources":["../../../../src/lib/slots/copilot-slot.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  Input,\n  TemplateRef,\n  ViewContainerRef,\n  OnInit,\n  OnChanges,\n  SimpleChanges,\n  Inject,\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\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  selector: 'copilot-slot',\n  standalone: true,\n  imports: [CommonModule],\n  template: `\n    <!-- If slot template provided, render it -->\n    <ng-container *ngIf=\"slot && isTemplate(slot)\"\n                  [ngTemplateOutlet]=\"slot\"\n                  [ngTemplateOutletContext]=\"context || {}\">\n    </ng-container>\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    <ng-content *ngIf=\"!slot && !defaultComponent\"></ng-content>\n  `,\n  changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class CopilotSlotComponent implements OnInit, OnChanges {\n  @Input() slot?: TemplateRef<any> | Type<any>;\n  @Input() context?: any;\n  @Input() defaultComponent?: Type<any>;\n  @Input() outputs?: Record<string, (event: any) => void>;\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\n    if (props) {\n      for (const key in props) {\n        const value = props[key];\n        this.componentRef.setInput(key, value);\n      }\n    }\n    \n    // Trigger change detection\n    if (this.componentRef.changeDetectorRef) {\n      this.componentRef.changeDetectorRef.detectChanges();\n    }\n  }\n}"]}
@@ -0,0 +1,6 @@
1
+ import { InjectionToken } from '@angular/core';
2
+ /**
3
+ * Injection token for slot configuration
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=
@@ -0,0 +1,222 @@
1
+ import { TemplateRef, inject } from '@angular/core';
2
+ import { SLOT_CONFIG } from './slot.types';
3
+ /**
4
+ * Renders a slot value into a ViewContainerRef.
5
+ * This is the core utility for slot rendering.
6
+ *
7
+ * @param viewContainer - The ViewContainerRef to render into
8
+ * @param options - Options for rendering the slot
9
+ * @returns The created component or embedded view reference
10
+ *
11
+ * @example
12
+ * ```typescript
13
+ * export class MyComponent {
14
+ * @ViewChild('container', { read: ViewContainerRef }) container!: ViewContainerRef;
15
+ *
16
+ * renderButton() {
17
+ * renderSlot(this.container, {
18
+ * slot: this.buttonOverride,
19
+ * defaultComponent: DefaultButton,
20
+ * props: { text: 'Click me' },
21
+ * outputs: { click: (event) => this.handleClick(event) }
22
+ * });
23
+ * }
24
+ * }
25
+ * ```
26
+ */
27
+ export function renderSlot(viewContainer, options) {
28
+ const { slot, defaultComponent, props, injector, outputs } = options;
29
+ viewContainer.clear();
30
+ const effectiveSlot = slot ?? defaultComponent;
31
+ const effectiveInjector = injector ?? viewContainer.injector;
32
+ if (effectiveSlot instanceof TemplateRef) {
33
+ // TemplateRef: render template
34
+ return viewContainer.createEmbeddedView(effectiveSlot, {
35
+ $implicit: props ?? {},
36
+ props: props ?? {}
37
+ });
38
+ }
39
+ else if (isComponentType(effectiveSlot)) {
40
+ // Component type - wrap in try/catch for safety
41
+ try {
42
+ return createComponent(viewContainer, effectiveSlot, props, effectiveInjector, outputs);
43
+ }
44
+ catch (error) {
45
+ console.warn('Failed to create component:', effectiveSlot, error);
46
+ // Fall through to default component
47
+ }
48
+ }
49
+ // Default: render default component if provided
50
+ return defaultComponent ? createComponent(viewContainer, defaultComponent, props, effectiveInjector, outputs) : null;
51
+ }
52
+ /**
53
+ * Creates a component and applies properties.
54
+ */
55
+ function createComponent(viewContainer, component, props, injector, outputs) {
56
+ const componentRef = viewContainer.createComponent(component, {
57
+ injector
58
+ });
59
+ if (props) {
60
+ // Apply props using setInput
61
+ for (const key in props) {
62
+ const value = props[key];
63
+ componentRef.setInput(key, value);
64
+ }
65
+ }
66
+ if (outputs) {
67
+ // Wire up output event handlers with proper cleanup
68
+ const instance = componentRef.instance;
69
+ const subscriptions = [];
70
+ for (const [eventName, handler] of Object.entries(outputs)) {
71
+ if (instance[eventName]?.subscribe) {
72
+ const subscription = instance[eventName].subscribe(handler);
73
+ subscriptions.push(subscription);
74
+ }
75
+ }
76
+ // Register cleanup on component destroy
77
+ componentRef.onDestroy(() => {
78
+ subscriptions.forEach(sub => sub.unsubscribe());
79
+ });
80
+ }
81
+ // Trigger change detection
82
+ componentRef.changeDetectorRef.detectChanges();
83
+ return componentRef;
84
+ }
85
+ /**
86
+ * Checks if a value is a component type.
87
+ * Simplified check - rely on try/catch for actual validation.
88
+ */
89
+ export function isComponentType(value) {
90
+ // Arrow functions and regular functions without a prototype are not components
91
+ return typeof value === 'function' && !!value.prototype;
92
+ }
93
+ /**
94
+ * Checks if a value is a valid slot value.
95
+ */
96
+ export function isSlotValue(value) {
97
+ return value instanceof TemplateRef || isComponentType(value);
98
+ }
99
+ /**
100
+ * Normalizes a slot value to a consistent format.
101
+ */
102
+ export function normalizeSlotValue(value, defaultComponent) {
103
+ if (!value) {
104
+ return { component: defaultComponent };
105
+ }
106
+ if (value instanceof TemplateRef) {
107
+ return { template: value };
108
+ }
109
+ if (isComponentType(value)) {
110
+ return { component: value };
111
+ }
112
+ return { component: defaultComponent };
113
+ }
114
+ /**
115
+ * Creates a slot configuration map for a component.
116
+ *
117
+ * @example
118
+ * ```typescript
119
+ * const slots = createSlotConfig({
120
+ * sendButton: CustomSendButton,
121
+ * toolbar: 'custom-toolbar-class',
122
+ * footer: footerTemplate
123
+ * }, {
124
+ * sendButton: DefaultSendButton,
125
+ * toolbar: DefaultToolbar,
126
+ * footer: DefaultFooter
127
+ * });
128
+ * ```
129
+ */
130
+ export function createSlotConfig(overrides, defaults) {
131
+ const config = new Map();
132
+ for (const key in defaults) {
133
+ const override = overrides[key];
134
+ const defaultComponent = defaults[key];
135
+ config.set(key, normalizeSlotValue(override, defaultComponent));
136
+ }
137
+ return config;
138
+ }
139
+ /**
140
+ * Provides slot configuration to child components via DI.
141
+ *
142
+ * @example
143
+ * ```typescript
144
+ * @Component({
145
+ * providers: [
146
+ * provideSlots({
147
+ * sendButton: CustomSendButton,
148
+ * toolbar: CustomToolbar
149
+ * })
150
+ * ]
151
+ * })
152
+ * ```
153
+ */
154
+ export function provideSlots(slots) {
155
+ const slotMap = new Map();
156
+ // Only accept component types in DI (templates lack view context)
157
+ for (const [key, value] of Object.entries(slots)) {
158
+ if (isComponentType(value)) {
159
+ slotMap.set(key, { component: value });
160
+ }
161
+ }
162
+ return {
163
+ provide: SLOT_CONFIG,
164
+ useValue: slotMap
165
+ };
166
+ }
167
+ /**
168
+ * Gets slot configuration from DI.
169
+ * Must be called within an injection context.
170
+ *
171
+ * @example
172
+ * ```typescript
173
+ * export class MyComponent {
174
+ * slots = getSlotConfig();
175
+ *
176
+ * ngOnInit() {
177
+ * const sendButton = this.slots?.get('sendButton');
178
+ * }
179
+ * }
180
+ * ```
181
+ */
182
+ export function getSlotConfig() {
183
+ return inject(SLOT_CONFIG, { optional: true });
184
+ }
185
+ /**
186
+ * Creates a render function for a specific slot.
187
+ * Useful for creating reusable slot renderers.
188
+ *
189
+ * @example
190
+ * ```typescript
191
+ * const renderSendButton = createSlotRenderer(
192
+ * DefaultSendButton,
193
+ * 'sendButton'
194
+ * );
195
+ *
196
+ * // Later in template
197
+ * renderSendButton(this.viewContainer, this.sendButtonOverride);
198
+ * ```
199
+ */
200
+ export function createSlotRenderer(defaultComponent, slotName) {
201
+ // Get config in the injection context when the renderer is created
202
+ const config = slotName ? getSlotConfig() : null;
203
+ return (viewContainer, slot, props, outputs) => {
204
+ // Check DI for overrides if slot name provided
205
+ if (slotName && !slot && config) {
206
+ const entry = config.get(slotName);
207
+ if (entry) {
208
+ if (entry.component)
209
+ slot = entry.component;
210
+ else if (entry.template)
211
+ slot = entry.template;
212
+ }
213
+ }
214
+ return renderSlot(viewContainer, {
215
+ slot,
216
+ defaultComponent,
217
+ props,
218
+ outputs
219
+ });
220
+ };
221
+ }
222
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"slot.utils.js","sourceRoot":"","sources":["../../../../src/lib/slots/slot.utils.ts"],"names":[],"mappings":"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","sourcesContent":["import { \n  Type, \n  TemplateRef, \n  ViewContainerRef,\n  ComponentRef,\n  EmbeddedViewRef,\n  Injector,\n  inject\n} from '@angular/core';\nimport { SlotValue, RenderSlotOptions, SlotRegistryEntry, SLOT_CONFIG } from './slot.types';\n\n/**\n * Renders a slot value into a ViewContainerRef.\n * This is the core utility for slot rendering.\n * \n * @param viewContainer - The ViewContainerRef to render into\n * @param options - Options for rendering the slot\n * @returns The created component or embedded view reference\n * \n * @example\n * ```typescript\n * export class MyComponent {\n *   @ViewChild('container', { read: ViewContainerRef }) container!: ViewContainerRef;\n *   \n *   renderButton() {\n *     renderSlot(this.container, {\n *       slot: this.buttonOverride,\n *       defaultComponent: DefaultButton,\n *       props: { text: 'Click me' },\n *       outputs: { click: (event) => this.handleClick(event) }\n *     });\n *   }\n * }\n * ```\n */\nexport function renderSlot<T = any>(\n  viewContainer: ViewContainerRef,\n  options: RenderSlotOptions<T>\n): ComponentRef<T> | EmbeddedViewRef<T> | null {\n  const { slot, defaultComponent, props, injector, outputs } = options;\n  \n  viewContainer.clear();\n  \n  const effectiveSlot = slot ?? defaultComponent;\n  const effectiveInjector = injector ?? viewContainer.injector;\n  \n  if (effectiveSlot instanceof TemplateRef) {\n    // TemplateRef: render template\n    return viewContainer.createEmbeddedView(effectiveSlot, {\n      $implicit: props ?? {},\n      props: props ?? {}\n    } as any);\n  } else if (isComponentType(effectiveSlot)) {\n    // Component type - wrap in try/catch for safety\n    try {\n      return createComponent(\n        viewContainer,\n        effectiveSlot as Type<T>,\n        props,\n        effectiveInjector,\n        outputs\n      );\n    } catch (error) {\n      console.warn('Failed to create component:', effectiveSlot, error);\n      // Fall through to default component\n    }\n  }\n  \n  // Default: render default component if provided\n  return defaultComponent ? createComponent(\n    viewContainer,\n    defaultComponent,\n    props,\n    effectiveInjector,\n    outputs\n  ) : null;\n}\n\n/**\n * Creates a component and applies properties.\n */\nfunction createComponent<T>(\n  viewContainer: ViewContainerRef,\n  component: Type<T>,\n  props?: Partial<T>,\n  injector?: Injector,\n  outputs?: Record<string, (event: any) => void>\n): ComponentRef<T> {\n  const componentRef = viewContainer.createComponent(component, {\n    injector\n  });\n  \n  if (props) {\n    // Apply props using setInput\n    for (const key in props) {\n      const value = props[key];\n      componentRef.setInput(key, value);\n    }\n  }\n  \n  if (outputs) {\n    // Wire up output event handlers with proper cleanup\n    const instance = componentRef.instance as any;\n    const subscriptions: any[] = [];\n    \n    for (const [eventName, handler] of Object.entries(outputs)) {\n      if (instance[eventName]?.subscribe) {\n        const subscription = instance[eventName].subscribe(handler);\n        subscriptions.push(subscription);\n      }\n    }\n    \n    // Register cleanup on component destroy\n    componentRef.onDestroy(() => {\n      subscriptions.forEach(sub => sub.unsubscribe());\n    });\n  }\n  \n  // Trigger change detection\n  componentRef.changeDetectorRef.detectChanges();\n  \n  return componentRef;\n}\n\n\n/**\n * Checks if a value is a component type.\n * Simplified check - rely on try/catch for actual validation.\n */\nexport function isComponentType(value: any): boolean {\n  // Arrow functions and regular functions without a prototype are not components\n  return typeof value === 'function' && !!value.prototype;\n}\n\n/**\n * Checks if a value is a valid slot value.\n */\nexport function isSlotValue(value: any): value is SlotValue {\n  return value instanceof TemplateRef || isComponentType(value);\n}\n\n/**\n * Normalizes a slot value to a consistent format.\n */\nexport function normalizeSlotValue<T = any>(\n  value: SlotValue<T> | undefined,\n  defaultComponent: Type<T> | undefined\n): SlotRegistryEntry<T> {\n  if (!value) {\n    return { component: defaultComponent };\n  }\n  \n  if (value instanceof TemplateRef) {\n    return { template: value };\n  }\n  \n  if (isComponentType(value)) {\n    return { component: value as Type<T> };\n  }\n  \n  return { component: defaultComponent };\n}\n\n/**\n * Creates a slot configuration map for a component.\n * \n * @example\n * ```typescript\n * const slots = createSlotConfig({\n *   sendButton: CustomSendButton,\n *   toolbar: 'custom-toolbar-class',\n *   footer: footerTemplate\n * }, {\n *   sendButton: DefaultSendButton,\n *   toolbar: DefaultToolbar,\n *   footer: DefaultFooter\n * });\n * ```\n */\nexport function createSlotConfig<T extends Record<string, Type<any>>>(\n  overrides: Partial<Record<keyof T, SlotValue>>,\n  defaults: T\n): Map<keyof T, SlotRegistryEntry> {\n  const config = new Map<keyof T, SlotRegistryEntry>();\n  \n  for (const key in defaults) {\n    const override = overrides[key];\n    const defaultComponent = defaults[key];\n    config.set(key, normalizeSlotValue(override, defaultComponent));\n  }\n  \n  return config;\n}\n\n/**\n * Provides slot configuration to child components via DI.\n * \n * @example\n * ```typescript\n * @Component({\n *   providers: [\n *     provideSlots({\n *       sendButton: CustomSendButton,\n *       toolbar: CustomToolbar\n *     })\n *   ]\n * })\n * ```\n */\nexport function provideSlots(slots: Record<string, Type<any>>) {\n  const slotMap = new Map<string, SlotRegistryEntry>();\n  \n  // Only accept component types in DI (templates lack view context)\n  for (const [key, value] of Object.entries(slots)) {\n    if (isComponentType(value)) {\n      slotMap.set(key, { component: value as Type<any> });\n    }\n  }\n  \n  return {\n    provide: SLOT_CONFIG,\n    useValue: slotMap\n  };\n}\n\n/**\n * Gets slot configuration from DI.\n * Must be called within an injection context.\n * \n * @example\n * ```typescript\n * export class MyComponent {\n *   slots = getSlotConfig();\n *   \n *   ngOnInit() {\n *     const sendButton = this.slots?.get('sendButton');\n *   }\n * }\n * ```\n */\nexport function getSlotConfig(): ReadonlyMap<string, SlotRegistryEntry> | null {\n  return inject(SLOT_CONFIG, { optional: true });\n}\n\n/**\n * Creates a render function for a specific slot.\n * Useful for creating reusable slot renderers.\n * \n * @example\n * ```typescript\n * const renderSendButton = createSlotRenderer(\n *   DefaultSendButton,\n *   'sendButton'\n * );\n * \n * // Later in template\n * renderSendButton(this.viewContainer, this.sendButtonOverride);\n * ```\n */\nexport function createSlotRenderer<T>(\n  defaultComponent: Type<T>,\n  slotName?: string\n) {\n  // Get config in the injection context when the renderer is created\n  const config = slotName ? getSlotConfig() : null;\n  \n  return (\n    viewContainer: ViewContainerRef,\n    slot?: SlotValue<T>,\n    props?: Partial<T>,\n    outputs?: Record<string, (event: any) => void>\n  ) => {\n    // Check DI for overrides if slot name provided\n    if (slotName && !slot && config) {\n      const entry = config.get(slotName);\n      if (entry) {\n        if (entry.component) slot = entry.component;\n        else if (entry.template) slot = entry.template;\n      }\n    }\n    \n    return renderSlot(viewContainer, {\n      slot,\n      defaultComponent,\n      props,\n      outputs\n    });\n  };\n}"]}
@@ -0,0 +1,10 @@
1
+ import { clsx } from 'clsx';
2
+ import { twMerge } from 'tailwind-merge';
3
+ /**
4
+ * Utility function to merge Tailwind CSS classes
5
+ * Combines clsx for conditional classes and tailwind-merge for proper Tailwind class merging
6
+ */
7
+ export function cn(...inputs) {
8
+ return twMerge(clsx(inputs));
9
+ }
10
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidXRpbHMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvbGliL3V0aWxzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxJQUFJLEVBQW1CLE1BQU0sTUFBTSxDQUFDO0FBQzdDLE9BQU8sRUFBRSxPQUFPLEVBQUUsTUFBTSxnQkFBZ0IsQ0FBQztBQUV6Qzs7O0dBR0c7QUFDSCxNQUFNLFVBQVUsRUFBRSxDQUFDLEdBQUcsTUFBb0I7SUFDeEMsT0FBTyxPQUFPLENBQUMsSUFBSSxDQUFDLE1BQU0sQ0FBQyxDQUFDLENBQUM7QUFDL0IsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IGNsc3gsIHR5cGUgQ2xhc3NWYWx1ZSB9IGZyb20gJ2Nsc3gnO1xuaW1wb3J0IHsgdHdNZXJnZSB9IGZyb20gJ3RhaWx3aW5kLW1lcmdlJztcblxuLyoqXG4gKiBVdGlsaXR5IGZ1bmN0aW9uIHRvIG1lcmdlIFRhaWx3aW5kIENTUyBjbGFzc2VzXG4gKiBDb21iaW5lcyBjbHN4IGZvciBjb25kaXRpb25hbCBjbGFzc2VzIGFuZCB0YWlsd2luZC1tZXJnZSBmb3IgcHJvcGVyIFRhaWx3aW5kIGNsYXNzIG1lcmdpbmdcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGNuKC4uLmlucHV0czogQ2xhc3NWYWx1ZVtdKSB7XG4gIHJldHVybiB0d01lcmdlKGNsc3goaW5wdXRzKSk7XG59Il19
@@ -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.service.js","sourceRoot":"","sources":["../../../src/services/resize-observer.service.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,UAAU,EAAqB,MAAM,eAAe,CAAC;AAC1E,OAAO,EAAc,OAAO,EAAE,eAAe,EAAE,MAAM,MAAM,CAAC;AAC5D,OAAO,EAAE,YAAY,EAAE,SAAS,EAAE,oBAAoB,EAAE,MAAM,gBAAgB,CAAC;;AAW/E,MAAM,OAAO,qBAAqB;IAMZ;IALZ,QAAQ,GAAG,IAAI,OAAO,EAAQ,CAAC;IAC/B,SAAS,GAAG,IAAI,GAAG,EAA+B,CAAC;IACnD,YAAY,GAAG,IAAI,GAAG,EAA6C,CAAC;IACpE,cAAc,GAAG,IAAI,GAAG,EAAuB,CAAC;IAExD,YAAoB,MAAc;QAAd,WAAM,GAAN,MAAM,CAAQ;IAAG,CAAC;IAEtC;;;;;OAKG;IACH,cAAc,CACZ,OAA8C,EAC9C,aAAqB,CAAC,EACtB,qBAA6B,GAAG;QAEhC,MAAM,EAAE,GAAG,OAAO,YAAY,UAAU,CAAC,CAAC,CAAC,OAAO,CAAC,aAAa,CAAC,CAAC,CAAC,OAAO,CAAC;QAE3E,gDAAgD;QAChD,IAAI,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,EAAE,CAAC,EAAE,CAAC;YAC9B,OAAO,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,EAAE,CAAE,CAAC,YAAY,EAAE,CAAC;QACnD,CAAC;QAED,sCAAsC;QACtC,MAAM,YAAY,GAAG,IAAI,eAAe,CAAc;YACpD,KAAK,EAAE,EAAE,CAAC,WAAW;YACrB,MAAM,EAAE,EAAE,CAAC,YAAY;YACvB,UAAU,EAAE,KAAK;SAClB,CAAC,CAAC;QAEH,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,EAAE,EAAE,YAAY,CAAC,CAAC;QAExC,wBAAwB;QACxB,MAAM,cAAc,GAAG,IAAI,cAAc,CAAC,CAAC,OAAO,EAAE,EAAE;YACpD,IAAI,OAAO,CAAC,MAAM,KAAK,CAAC;gBAAE,OAAO;YAEjC,MAAM,KAAK,GAAG,OAAO,CAAC,CAAC,CAAC,CAAC;YACzB,IAAI,CAAC,KAAK;gBAAE,OAAO;YAEnB,MAAM,EAAE,KAAK,EAAE,MAAM,EAAE,GAAG,KAAK,CAAC,WAAW,CAAC;YAE5C,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,GAAG,EAAE;gBACnB,yBAAyB;gBACzB,MAAM,eAAe,GAAG,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;gBACpD,IAAI,eAAe,EAAE,CAAC;oBACpB,YAAY,CAAC,eAAe,CAAC,CAAC;gBAChC,CAAC;gBAED,sCAAsC;gBACtC,YAAY,CAAC,IAAI,CAAC;oBAChB,KAAK;oBACL,MAAM;oBACN,UAAU,EAAE,IAAI;iBACjB,CAAC,CAAC;gBAEH,uCAAuC;gBACvC,IAAI,kBAAkB,GAAG,CAAC,EAAE,CAAC;oBAC3B,MAAM,OAAO,GAAG,MAAM,CAAC,UAAU,CAAC,GAAG,EAAE;wBACrC,YAAY,CAAC,IAAI,CAAC;4BAChB,KAAK;4BACL,MAAM;4BACN,UAAU,EAAE,KAAK;yBAClB,CAAC,CAAC;wBACH,IAAI,CAAC,cAAc,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC;oBACjC,CAAC,EAAE,kBAAkB,CAAC,CAAC;oBAEvB,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,EAAE,EAAE,OAAO,CAAC,CAAC;gBACvC,CAAC;qBAAM,CAAC;oBACN,sDAAsD;oBACtD,YAAY,CAAC,IAAI,CAAC;wBAChB,KAAK;wBACL,MAAM;wBACN,UAAU,EAAE,KAAK;qBAClB,CAAC,CAAC;gBACL,CAAC;YACH,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,kBAAkB;QAClB,cAAc,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC;QAC3B,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,EAAE,EAAE,cAAc,CAAC,CAAC;QAEvC,iDAAiD;QACjD,MAAM,UAAU,GAAG,YAAY,CAAC,YAAY,EAAE,CAAC,IAAI,CACjD,UAAU,GAAG,CAAC,CAAC,CAAC,CAAC,YAAY,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,CAAC,MAAM,EAAE,EAAE,CAAC,MAAM,EAC9D,oBAAoB,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAC5B,CAAC,CAAC,KAAK,KAAK,CAAC,CAAC,KAAK;YACnB,CAAC,CAAC,MAAM,KAAK,CAAC,CAAC,MAAM;YACrB,CAAC,CAAC,UAAU,KAAK,CAAC,CAAC,UAAU,CAC9B,EACD,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,CACzB,CAAC;QAEF,OAAO,UAAU,CAAC;IACpB,CAAC;IAED;;;OAGG;IACH,SAAS,CAAC,OAA8C;QACtD,MAAM,EAAE,GAAG,OAAO,YAAY,UAAU,CAAC,CAAC,CAAC,OAAO,CAAC,aAAa,CAAC,CAAC,CAAC,OAAO,CAAC;QAE3E,0BAA0B;QAC1B,MAAM,OAAO,GAAG,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;QAC5C,IAAI,OAAO,EAAE,CAAC;YACZ,YAAY,CAAC,OAAO,CAAC,CAAC;YACtB,IAAI,CAAC,cAAc,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC;QACjC,CAAC;QAED,sBAAsB;QACtB,MAAM,QAAQ,GAAG,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;QACxC,IAAI,QAAQ,EAAE,CAAC;YACb,QAAQ,CAAC,UAAU,EAAE,CAAC;YACtB,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC;QAC5B,CAAC;QAED,8BAA8B;QAC9B,MAAM,OAAO,GAAG,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;QAC1C,IAAI,OAAO,EAAE,CAAC;YACZ,OAAO,CAAC,QAAQ,EAAE,CAAC;YACnB,IAAI,CAAC,YAAY,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC;QAC/B,CAAC;IACH,CAAC;IAED;;;OAGG;IACH,cAAc,CAAC,OAA8C;QAC3D,MAAM,EAAE,GAAG,OAAO,YAAY,UAAU,CAAC,CAAC,CAAC,OAAO,CAAC,aAAa,CAAC,CAAC,CAAC,OAAO,CAAC;QAC3E,OAAO;YACL,KAAK,EAAE,EAAE,CAAC,WAAW;YACrB,MAAM,EAAE,EAAE,CAAC,YAAY;SACxB,CAAC;IACJ,CAAC;IAED;;;OAGG;IACH,eAAe,CAAC,OAA8C;QAC5D,MAAM,EAAE,GAAG,OAAO,YAAY,UAAU,CAAC,CAAC,CAAC,OAAO,CAAC,aAAa,CAAC,CAAC,CAAC,OAAO,CAAC;QAC3E,MAAM,OAAO,GAAG,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;QAC1C,OAAO,OAAO,CAAC,CAAC,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,CAAC,IAAI,CAAC;IACxC,CAAC;IAED,WAAW;QACT,qBAAqB;QACrB,IAAI,CAAC,cAAc,CAAC,OAAO,CAAC,OAAO,CAAC,EAAE,CAAC,YAAY,CAAC,OAAO,CAAC,CAAC,CAAC;QAC9D,IAAI,CAAC,cAAc,CAAC,KAAK,EAAE,CAAC;QAE5B,2BAA2B;QAC3B,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC,QAAQ,CAAC,EAAE,CAAC,QAAQ,CAAC,UAAU,EAAE,CAAC,CAAC;QAC1D,IAAI,CAAC,SAAS,CAAC,KAAK,EAAE,CAAC;QAEvB,wBAAwB;QACxB,IAAI,CAAC,YAAY,CAAC,OAAO,CAAC,OAAO,CAAC,EAAE,CAAC,OAAO,CAAC,QAAQ,EAAE,CAAC,CAAC;QACzD,IAAI,CAAC,YAAY,CAAC,KAAK,EAAE,CAAC;QAE1B,2BAA2B;QAC3B,IAAI,CAAC,QAAQ,CAAC,IAAI,EAAE,CAAC;QACrB,IAAI,CAAC,QAAQ,CAAC,QAAQ,EAAE,CAAC;IAC3B,CAAC;wGAtKU,qBAAqB;4GAArB,qBAAqB,cAFpB,MAAM;;4FAEP,qBAAqB;kBAHjC,UAAU;mBAAC;oBACV,UAAU,EAAE,MAAM;iBACnB","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((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): { width: number; height: number } {\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(element: ElementRef<HTMLElement> | HTMLElement): 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}"]}