@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,10 @@
1
+ /**
2
+ * Error class for audio recorder failures
3
+ */
4
+ export class AudioRecorderError extends Error {
5
+ constructor(message) {
6
+ super(message);
7
+ this.name = 'AudioRecorderError';
8
+ }
9
+ }
10
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29waWxvdC1jaGF0LWlucHV0LnR5cGVzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vc3JjL2NvbXBvbmVudHMvY2hhdC9jb3BpbG90LWNoYXQtaW5wdXQudHlwZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBNEJBOztHQUVHO0FBQ0gsTUFBTSxPQUFPLGtCQUFtQixTQUFRLEtBQUs7SUFDM0MsWUFBWSxPQUFlO1FBQ3pCLEtBQUssQ0FBQyxPQUFPLENBQUMsQ0FBQztRQUNmLElBQUksQ0FBQyxJQUFJLEdBQUcsb0JBQW9CLENBQUM7SUFDbkMsQ0FBQztDQUNGIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHR5cGUgeyBUeXBlLCBUZW1wbGF0ZVJlZiB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuXG4vKipcbiAqIE1vZGUgb2YgdGhlIGNoYXQgaW5wdXQgY29tcG9uZW50XG4gKi9cbmV4cG9ydCB0eXBlIENvcGlsb3RDaGF0SW5wdXRNb2RlID0gJ2lucHV0JyB8ICd0cmFuc2NyaWJlJyB8ICdwcm9jZXNzaW5nJztcblxuLyoqXG4gKiBSZXByZXNlbnRzIGEgbWVudSBpdGVtIGluIHRoZSB0b29scyBtZW51XG4gKi9cbmV4cG9ydCB0eXBlIFRvb2xzTWVudUl0ZW0gPSB7XG4gIGxhYmVsOiBzdHJpbmc7XG59ICYgKFxuICB8IHtcbiAgICAgIGFjdGlvbjogKCkgPT4gdm9pZDtcbiAgICAgIGl0ZW1zPzogbmV2ZXI7XG4gICAgfVxuICB8IHtcbiAgICAgIGFjdGlvbj86IG5ldmVyO1xuICAgICAgaXRlbXM6IChUb29sc01lbnVJdGVtIHwgJy0nKVtdO1xuICAgIH1cbik7XG5cbi8qKlxuICogQXVkaW8gcmVjb3JkZXIgc3RhdGVcbiAqL1xuZXhwb3J0IHR5cGUgQXVkaW9SZWNvcmRlclN0YXRlID0gJ2lkbGUnIHwgJ3JlY29yZGluZycgfCAncHJvY2Vzc2luZyc7XG5cbi8qKlxuICogRXJyb3IgY2xhc3MgZm9yIGF1ZGlvIHJlY29yZGVyIGZhaWx1cmVzXG4gKi9cbmV4cG9ydCBjbGFzcyBBdWRpb1JlY29yZGVyRXJyb3IgZXh0ZW5kcyBFcnJvciB7XG4gIGNvbnN0cnVjdG9yKG1lc3NhZ2U6IHN0cmluZykge1xuICAgIHN1cGVyKG1lc3NhZ2UpO1xuICAgIHRoaXMubmFtZSA9ICdBdWRpb1JlY29yZGVyRXJyb3InO1xuICB9XG59XG5cbi8qKlxuICogUHJvcHMgZm9yIHRleHRhcmVhIGNvbXBvbmVudFxuICovXG5leHBvcnQgaW50ZXJmYWNlIENvcGlsb3RDaGF0VGV4dGFyZWFQcm9wcyB7XG4gIHZhbHVlPzogc3RyaW5nO1xuICBwbGFjZWhvbGRlcj86IHN0cmluZztcbiAgbWF4Um93cz86IG51bWJlcjtcbiAgYXV0b0ZvY3VzPzogYm9vbGVhbjtcbiAgZGlzYWJsZWQ/OiBib29sZWFuO1xuICBvbkNoYW5nZT86ICh2YWx1ZTogc3RyaW5nKSA9PiB2b2lkO1xuICBvbktleURvd24/OiAoZXZlbnQ6IEtleWJvYXJkRXZlbnQpID0+IHZvaWQ7XG4gIGlucHV0Q2xhc3M/OiBzdHJpbmc7XG4gIHN0eWxlPzogYW55O1xuICByb3dzPzogbnVtYmVyO1xuICBjb2xzPzogbnVtYmVyO1xuICByZWFkb25seT86IGJvb2xlYW47XG4gIHNwZWxsY2hlY2s/OiBib29sZWFuO1xuICB3cmFwPzogJ2hhcmQnIHwgJ3NvZnQnIHwgJ29mZic7XG59XG5cbi8qKlxuICogUHJvcHMgZm9yIGJ1dHRvbiBjb21wb25lbnRzXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgQ29waWxvdENoYXRCdXR0b25Qcm9wcyB7XG4gIGRpc2FibGVkPzogYm9vbGVhbjtcbiAgb25DbGljaz86ICgpID0+IHZvaWQ7XG4gIGlucHV0Q2xhc3M/OiBzdHJpbmc7XG4gIHN0eWxlPzogYW55O1xuICB0eXBlPzogJ2J1dHRvbicgfCAnc3VibWl0JyB8ICdyZXNldCc7XG4gIGFyaWFMYWJlbD86IHN0cmluZztcbiAgYXJpYVByZXNzZWQ/OiBib29sZWFuO1xuICBhcmlhRXhwYW5kZWQ/OiBib29sZWFuO1xuICB0aXRsZT86IHN0cmluZztcbn1cblxuLyoqXG4gKiBQcm9wcyBmb3IgdG9vbGJhciBidXR0b24gd2l0aCB0b29sdGlwXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgQ29waWxvdENoYXRUb29sYmFyQnV0dG9uUHJvcHMgZXh0ZW5kcyBDb3BpbG90Q2hhdEJ1dHRvblByb3BzIHtcbiAgaWNvbj86IFRlbXBsYXRlUmVmPGFueT47XG4gIHRvb2x0aXA/OiBzdHJpbmc7XG4gIHZhcmlhbnQ/OiAncHJpbWFyeScgfCAnc2Vjb25kYXJ5Jztcbn1cblxuLyoqXG4gKiBQcm9wcyBmb3IgdG9vbHMgbWVudSBidXR0b25cbiAqL1xuZXhwb3J0IGludGVyZmFjZSBDb3BpbG90Q2hhdFRvb2xzQnV0dG9uUHJvcHMgZXh0ZW5kcyBDb3BpbG90Q2hhdEJ1dHRvblByb3BzIHtcbiAgdG9vbHNNZW51PzogKFRvb2xzTWVudUl0ZW0gfCAnLScpW107XG59XG5cbi8qKlxuICogUHJvcHMgZm9yIGF1ZGlvIHJlY29yZGVyXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgQ29waWxvdENoYXRBdWRpb1JlY29yZGVyUHJvcHMge1xuICBpbnB1dENsYXNzPzogc3RyaW5nO1xuICBzdHlsZT86IGFueTtcbiAgb25TdGF0ZUNoYW5nZT86IChzdGF0ZTogQXVkaW9SZWNvcmRlclN0YXRlKSA9PiB2b2lkO1xuICBzaG93Q29udHJvbHM/OiBib29sZWFuO1xuICBtYXhEdXJhdGlvbj86IG51bWJlcjtcbn1cblxuLyoqXG4gKiBQcm9wcyBmb3IgdG9vbGJhclxuICovXG5leHBvcnQgaW50ZXJmYWNlIENvcGlsb3RDaGF0VG9vbGJhclByb3BzIHtcbiAgaW5wdXRDbGFzcz86IHN0cmluZztcbiAgc3R5bGU/OiBhbnk7XG4gIHBvc2l0aW9uPzogJ3RvcCcgfCAnYm90dG9tJztcbiAgYWxpZ25tZW50PzogJ2xlZnQnIHwgJ2NlbnRlcicgfCAncmlnaHQnIHwgJ3NwYWNlLWJldHdlZW4nO1xufVxuXG4vKipcbiAqIFNsb3QgY29uZmlndXJhdGlvbiBmb3IgY2hhdCBpbnB1dFxuICovXG5leHBvcnQgaW50ZXJmYWNlIENvcGlsb3RDaGF0SW5wdXRTbG90cyB7XG4gIHRleHRBcmVhPzogVHlwZTxhbnk+IHwgVGVtcGxhdGVSZWY8YW55PjtcbiAgc2VuZEJ1dHRvbj86IFR5cGU8YW55PiB8IFRlbXBsYXRlUmVmPGFueT47XG4gIHN0YXJ0VHJhbnNjcmliZUJ1dHRvbj86IFR5cGU8YW55PiB8IFRlbXBsYXRlUmVmPGFueT47XG4gIGNhbmNlbFRyYW5zY3JpYmVCdXR0b24/OiBUeXBlPGFueT4gfCBUZW1wbGF0ZVJlZjxhbnk+O1xuICBmaW5pc2hUcmFuc2NyaWJlQnV0dG9uPzogVHlwZTxhbnk+IHwgVGVtcGxhdGVSZWY8YW55PjtcbiAgYWRkRmlsZUJ1dHRvbj86IFR5cGU8YW55PiB8IFRlbXBsYXRlUmVmPGFueT47XG4gIHRvb2xzQnV0dG9uPzogVHlwZTxhbnk+IHwgVGVtcGxhdGVSZWY8YW55PjtcbiAgdG9vbGJhcj86IFR5cGU8YW55PiB8IFRlbXBsYXRlUmVmPGFueT47XG4gIGF1ZGlvUmVjb3JkZXI/OiBUeXBlPGFueT4gfCBUZW1wbGF0ZVJlZjxhbnk+O1xufVxuXG4vKipcbiAqIElucHV0IGNvbmZpZ3VyYXRpb24gZm9yIHRoZSBjaGF0IGlucHV0IGNvbXBvbmVudFxuICovXG5leHBvcnQgaW50ZXJmYWNlIENvcGlsb3RDaGF0SW5wdXRDb25maWcge1xuICBtb2RlPzogQ29waWxvdENoYXRJbnB1dE1vZGU7XG4gIHRvb2xzTWVudT86IChUb29sc01lbnVJdGVtIHwgJy0nKVtdO1xuICBhdXRvRm9jdXM/OiBib29sZWFuO1xuICBhZGRpdGlvbmFsVG9vbGJhckl0ZW1zPzogVGVtcGxhdGVSZWY8YW55PjtcbiAgdmFsdWU/OiBzdHJpbmc7XG4gIGNsYXNzPzogc3RyaW5nO1xufVxuXG4vKipcbiAqIE91dHB1dCBldmVudHMgZm9yIHRoZSBjaGF0IGlucHV0IGNvbXBvbmVudFxuICovXG5leHBvcnQgaW50ZXJmYWNlIENvcGlsb3RDaGF0SW5wdXRPdXRwdXRzIHtcbiAgc3VibWl0TWVzc2FnZTogKHZhbHVlOiBzdHJpbmcpID0+IHZvaWQ7XG4gIHN0YXJ0VHJhbnNjcmliZTogKCkgPT4gdm9pZDtcbiAgY2FuY2VsVHJhbnNjcmliZTogKCkgPT4gdm9pZDtcbiAgZmluaXNoVHJhbnNjcmliZTogKCkgPT4gdm9pZDtcbiAgYWRkRmlsZTogKCkgPT4gdm9pZDtcbiAgY2hhbmdlVmFsdWU6ICh2YWx1ZTogc3RyaW5nKSA9PiB2b2lkO1xufVxuIl19
@@ -0,0 +1,45 @@
1
+ import { Component, Input, ChangeDetectionStrategy, ViewEncapsulation, computed, signal } from '@angular/core';
2
+ import { CommonModule } from '@angular/common';
3
+ import { cn } from '../../lib/utils';
4
+ import * as i0 from "@angular/core";
5
+ /**
6
+ * Cursor component that matches the React implementation exactly.
7
+ * Shows a pulsing dot animation to indicate activity.
8
+ */
9
+ export class CopilotChatMessageViewCursorComponent {
10
+ inputClass;
11
+ // Signal for reactive class updates
12
+ inputClassSignal = signal(undefined);
13
+ // Computed class that matches React exactly: w-[11px] h-[11px] rounded-full bg-foreground animate-pulse-cursor ml-1
14
+ computedClass = computed(() => cn('w-[11px] h-[11px] rounded-full bg-foreground animate-pulse-cursor ml-1', this.inputClassSignal()));
15
+ ngOnInit() {
16
+ this.inputClassSignal.set(this.inputClass);
17
+ }
18
+ ngOnChanges() {
19
+ this.inputClassSignal.set(this.inputClass);
20
+ }
21
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CopilotChatMessageViewCursorComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
22
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: CopilotChatMessageViewCursorComponent, isStandalone: true, selector: "copilot-chat-message-view-cursor", inputs: { inputClass: "inputClass" }, usesOnChanges: true, ngImport: i0, template: `
23
+ <div
24
+ [class]="computedClass()"
25
+ ></div>
26
+ `, isInline: true, dependencies: [{ kind: "ngmodule", type: CommonModule }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None });
27
+ }
28
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CopilotChatMessageViewCursorComponent, decorators: [{
29
+ type: Component,
30
+ args: [{
31
+ selector: 'copilot-chat-message-view-cursor',
32
+ standalone: true,
33
+ imports: [CommonModule],
34
+ changeDetection: ChangeDetectionStrategy.OnPush,
35
+ encapsulation: ViewEncapsulation.None,
36
+ template: `
37
+ <div
38
+ [class]="computedClass()"
39
+ ></div>
40
+ `
41
+ }]
42
+ }], propDecorators: { inputClass: [{
43
+ type: Input
44
+ }] } });
45
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29waWxvdC1jaGF0LW1lc3NhZ2Utdmlldy1jdXJzb3IuY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vc3JjL2NvbXBvbmVudHMvY2hhdC9jb3BpbG90LWNoYXQtbWVzc2FnZS12aWV3LWN1cnNvci5jb21wb25lbnQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUNMLFNBQVMsRUFDVCxLQUFLLEVBQ0wsdUJBQXVCLEVBQ3ZCLGlCQUFpQixFQUNqQixRQUFRLEVBQ1IsTUFBTSxFQUdQLE1BQU0sZUFBZSxDQUFDO0FBQ3ZCLE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQUMvQyxPQUFPLEVBQUUsRUFBRSxFQUFFLE1BQU0saUJBQWlCLENBQUM7O0FBRXJDOzs7R0FHRztBQWFILE1BQU0sT0FBTyxxQ0FBcUM7SUFDdkMsVUFBVSxDQUFVO0lBRTdCLG9DQUFvQztJQUM1QixnQkFBZ0IsR0FBRyxNQUFNLENBQXFCLFNBQVMsQ0FBQyxDQUFDO0lBRWpFLG9IQUFvSDtJQUNwSCxhQUFhLEdBQUcsUUFBUSxDQUFDLEdBQUcsRUFBRSxDQUM1QixFQUFFLENBQ0Esd0VBQXdFLEVBQ3hFLElBQUksQ0FBQyxnQkFBZ0IsRUFBRSxDQUN4QixDQUNGLENBQUM7SUFFRixRQUFRO1FBQ04sSUFBSSxDQUFDLGdCQUFnQixDQUFDLEdBQUcsQ0FBQyxJQUFJLENBQUMsVUFBVSxDQUFDLENBQUM7SUFDN0MsQ0FBQztJQUVELFdBQVc7UUFDVCxJQUFJLENBQUMsZ0JBQWdCLENBQUMsR0FBRyxDQUFDLElBQUksQ0FBQyxVQUFVLENBQUMsQ0FBQztJQUM3QyxDQUFDO3dHQXBCVSxxQ0FBcUM7NEZBQXJDLHFDQUFxQyx1SkFOdEM7Ozs7R0FJVCwyREFQUyxZQUFZOzs0RkFTWCxxQ0FBcUM7a0JBWmpELFNBQVM7bUJBQUM7b0JBQ1QsUUFBUSxFQUFFLGtDQUFrQztvQkFDNUMsVUFBVSxFQUFFLElBQUk7b0JBQ2hCLE9BQU8sRUFBRSxDQUFDLFlBQVksQ0FBQztvQkFDdkIsZUFBZSxFQUFFLHVCQUF1QixDQUFDLE1BQU07b0JBQy9DLGFBQWEsRUFBRSxpQkFBaUIsQ0FBQyxJQUFJO29CQUNyQyxRQUFRLEVBQUU7Ozs7R0FJVDtpQkFDRjs4QkFFVSxVQUFVO3NCQUFsQixLQUFLIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHtcbiAgQ29tcG9uZW50LFxuICBJbnB1dCxcbiAgQ2hhbmdlRGV0ZWN0aW9uU3RyYXRlZ3ksXG4gIFZpZXdFbmNhcHN1bGF0aW9uLFxuICBjb21wdXRlZCxcbiAgc2lnbmFsLFxuICBPbkluaXQsXG4gIE9uQ2hhbmdlc1xufSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IENvbW1vbk1vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvbW1vbic7XG5pbXBvcnQgeyBjbiB9IGZyb20gJy4uLy4uL2xpYi91dGlscyc7XG5cbi8qKlxuICogQ3Vyc29yIGNvbXBvbmVudCB0aGF0IG1hdGNoZXMgdGhlIFJlYWN0IGltcGxlbWVudGF0aW9uIGV4YWN0bHkuXG4gKiBTaG93cyBhIHB1bHNpbmcgZG90IGFuaW1hdGlvbiB0byBpbmRpY2F0ZSBhY3Rpdml0eS5cbiAqL1xuQENvbXBvbmVudCh7XG4gIHNlbGVjdG9yOiAnY29waWxvdC1jaGF0LW1lc3NhZ2Utdmlldy1jdXJzb3InLFxuICBzdGFuZGFsb25lOiB0cnVlLFxuICBpbXBvcnRzOiBbQ29tbW9uTW9kdWxlXSxcbiAgY2hhbmdlRGV0ZWN0aW9uOiBDaGFuZ2VEZXRlY3Rpb25TdHJhdGVneS5PblB1c2gsXG4gIGVuY2Fwc3VsYXRpb246IFZpZXdFbmNhcHN1bGF0aW9uLk5vbmUsXG4gIHRlbXBsYXRlOiBgXG4gICAgPGRpdlxuICAgICAgW2NsYXNzXT1cImNvbXB1dGVkQ2xhc3MoKVwiXG4gICAgPjwvZGl2PlxuICBgXG59KVxuZXhwb3J0IGNsYXNzIENvcGlsb3RDaGF0TWVzc2FnZVZpZXdDdXJzb3JDb21wb25lbnQgaW1wbGVtZW50cyBPbkluaXQsIE9uQ2hhbmdlcyB7XG4gIEBJbnB1dCgpIGlucHV0Q2xhc3M/OiBzdHJpbmc7XG4gIFxuICAvLyBTaWduYWwgZm9yIHJlYWN0aXZlIGNsYXNzIHVwZGF0ZXNcbiAgcHJpdmF0ZSBpbnB1dENsYXNzU2lnbmFsID0gc2lnbmFsPHN0cmluZyB8IHVuZGVmaW5lZD4odW5kZWZpbmVkKTtcbiAgXG4gIC8vIENvbXB1dGVkIGNsYXNzIHRoYXQgbWF0Y2hlcyBSZWFjdCBleGFjdGx5OiB3LVsxMXB4XSBoLVsxMXB4XSByb3VuZGVkLWZ1bGwgYmctZm9yZWdyb3VuZCBhbmltYXRlLXB1bHNlLWN1cnNvciBtbC0xXG4gIGNvbXB1dGVkQ2xhc3MgPSBjb21wdXRlZCgoKSA9PiBcbiAgICBjbihcbiAgICAgICd3LVsxMXB4XSBoLVsxMXB4XSByb3VuZGVkLWZ1bGwgYmctZm9yZWdyb3VuZCBhbmltYXRlLXB1bHNlLWN1cnNvciBtbC0xJyxcbiAgICAgIHRoaXMuaW5wdXRDbGFzc1NpZ25hbCgpXG4gICAgKVxuICApO1xuICBcbiAgbmdPbkluaXQoKSB7XG4gICAgdGhpcy5pbnB1dENsYXNzU2lnbmFsLnNldCh0aGlzLmlucHV0Q2xhc3MpO1xuICB9XG4gIFxuICBuZ09uQ2hhbmdlcygpIHtcbiAgICB0aGlzLmlucHV0Q2xhc3NTaWduYWwuc2V0KHRoaXMuaW5wdXRDbGFzcyk7XG4gIH1cbn0iXX0=
@@ -0,0 +1,296 @@
1
+ import { Component, Input, Output, EventEmitter, ContentChild, ChangeDetectionStrategy, ViewEncapsulation, signal, computed } from '@angular/core';
2
+ import { CommonModule } from '@angular/common';
3
+ import { CopilotSlotComponent } from '../../lib/slots/copilot-slot.component';
4
+ import { CopilotChatAssistantMessageComponent } from './copilot-chat-assistant-message.component';
5
+ import { CopilotChatUserMessageComponent } from './copilot-chat-user-message.component';
6
+ import { CopilotChatMessageViewCursorComponent } from './copilot-chat-message-view-cursor.component';
7
+ import { cn } from '../../lib/utils';
8
+ import * as i0 from "@angular/core";
9
+ import * as i1 from "@angular/common";
10
+ /**
11
+ * CopilotChatMessageView component - Angular port of the React component.
12
+ * Renders a list of chat messages with support for custom slots and layouts.
13
+ * DOM structure and Tailwind classes match the React implementation exactly.
14
+ */
15
+ export class CopilotChatMessageViewComponent {
16
+ // Core inputs matching React props
17
+ messages = [];
18
+ showCursor = false;
19
+ isLoading = false;
20
+ inputClass;
21
+ // Handler availability handled via DI service
22
+ // Assistant message slot inputs
23
+ assistantMessageComponent;
24
+ assistantMessageTemplate;
25
+ assistantMessageClass;
26
+ // User message slot inputs
27
+ userMessageComponent;
28
+ userMessageTemplate;
29
+ userMessageClass;
30
+ // Cursor slot inputs
31
+ cursorComponent;
32
+ cursorTemplate;
33
+ cursorClass;
34
+ // Custom layout template (render prop pattern)
35
+ customLayoutTemplate;
36
+ // Output events (bubbled from child components)
37
+ assistantMessageThumbsUp = new EventEmitter();
38
+ assistantMessageThumbsDown = new EventEmitter();
39
+ assistantMessageReadAloud = new EventEmitter();
40
+ assistantMessageRegenerate = new EventEmitter();
41
+ userMessageCopy = new EventEmitter();
42
+ userMessageEdit = new EventEmitter();
43
+ // Default components for slots
44
+ defaultAssistantComponent = CopilotChatAssistantMessageComponent;
45
+ defaultUserComponent = CopilotChatUserMessageComponent;
46
+ defaultCursorComponent = CopilotChatMessageViewCursorComponent;
47
+ // Signals for reactive updates
48
+ messagesSignal = signal([]);
49
+ showCursorSignal = signal(false);
50
+ isLoadingSignal = signal(false);
51
+ inputClassSignal = signal(undefined);
52
+ // Computed class matching React: twMerge("flex flex-col", className)
53
+ computedClass = computed(() => cn('flex flex-col', this.inputClassSignal()));
54
+ // Layout context for custom templates (render prop pattern)
55
+ layoutContext = computed(() => ({
56
+ isLoading: this.isLoadingSignal(),
57
+ messages: this.messagesSignal(),
58
+ showCursor: this.showCursorSignal(),
59
+ messageElements: this.messagesSignal().filter(m => m && (m.role === 'assistant' || m.role === 'user'))
60
+ }));
61
+ // Slot resolution computed signals
62
+ assistantMessageSlot = computed(() => this.assistantMessageComponent || this.assistantMessageClass);
63
+ userMessageSlot = computed(() => this.userMessageComponent || this.userMessageClass);
64
+ cursorSlot = computed(() => this.cursorComponent || this.cursorClass);
65
+ // Props merging helpers
66
+ mergeAssistantProps(message) {
67
+ return {
68
+ message,
69
+ messages: this.messagesSignal(),
70
+ isLoading: this.isLoadingSignal(),
71
+ inputClass: this.assistantMessageClass
72
+ };
73
+ }
74
+ mergeUserProps(message) {
75
+ return {
76
+ message,
77
+ inputClass: this.userMessageClass
78
+ };
79
+ }
80
+ // TrackBy function for performance optimization
81
+ trackByMessageId(index, message) {
82
+ return message?.id || `index-${index}`;
83
+ }
84
+ // Lifecycle hooks
85
+ ngOnInit() {
86
+ // Initialize signals with input values
87
+ this.messagesSignal.set(this.messages);
88
+ this.showCursorSignal.set(this.showCursor);
89
+ this.isLoadingSignal.set(this.isLoading);
90
+ this.inputClassSignal.set(this.inputClass);
91
+ }
92
+ ngOnChanges() {
93
+ this.messagesSignal.set(this.messages);
94
+ this.showCursorSignal.set(this.showCursor);
95
+ this.isLoadingSignal.set(this.isLoading);
96
+ this.inputClassSignal.set(this.inputClass);
97
+ }
98
+ // Event handlers - just pass them through
99
+ handleAssistantThumbsUp(event) {
100
+ this.assistantMessageThumbsUp.emit(event);
101
+ }
102
+ handleAssistantThumbsDown(event) {
103
+ this.assistantMessageThumbsDown.emit(event);
104
+ }
105
+ handleAssistantReadAloud(event) {
106
+ this.assistantMessageReadAloud.emit(event);
107
+ }
108
+ handleAssistantRegenerate(event) {
109
+ this.assistantMessageRegenerate.emit(event);
110
+ }
111
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CopilotChatMessageViewComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
112
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: CopilotChatMessageViewComponent, isStandalone: true, selector: "copilot-chat-message-view", inputs: { messages: "messages", showCursor: "showCursor", isLoading: "isLoading", inputClass: "inputClass", assistantMessageComponent: "assistantMessageComponent", assistantMessageTemplate: "assistantMessageTemplate", assistantMessageClass: "assistantMessageClass", userMessageComponent: "userMessageComponent", userMessageTemplate: "userMessageTemplate", userMessageClass: "userMessageClass", cursorComponent: "cursorComponent", cursorTemplate: "cursorTemplate", cursorClass: "cursorClass" }, outputs: { assistantMessageThumbsUp: "assistantMessageThumbsUp", assistantMessageThumbsDown: "assistantMessageThumbsDown", assistantMessageReadAloud: "assistantMessageReadAloud", assistantMessageRegenerate: "assistantMessageRegenerate", userMessageCopy: "userMessageCopy", userMessageEdit: "userMessageEdit" }, queries: [{ propertyName: "customLayoutTemplate", first: true, predicate: ["customLayout"], descendants: true }], usesOnChanges: true, ngImport: i0, template: `
113
+ <!-- Custom layout template support (render prop pattern) -->
114
+ @if (customLayoutTemplate) {
115
+ <ng-container *ngTemplateOutlet="customLayoutTemplate; context: layoutContext()"></ng-container>
116
+ } @else {
117
+ <!-- Default layout - exact React DOM structure: div with "flex flex-col" classes -->
118
+ <div [class]="computedClass()">
119
+ <!-- Message iteration - simplified without tool calls -->
120
+ @for (message of messagesSignal(); track trackByMessageId($index, message)) {
121
+ @if (message && message.role === 'assistant') {
122
+ <!-- Assistant message with slot support -->
123
+ @if (assistantMessageComponent || assistantMessageTemplate) {
124
+ <copilot-slot
125
+ [slot]="assistantMessageTemplate || assistantMessageComponent"
126
+ [context]="mergeAssistantProps(message)"
127
+ [defaultComponent]="defaultAssistantComponent">
128
+ </copilot-slot>
129
+ } @else {
130
+ <copilot-chat-assistant-message
131
+ [message]="message"
132
+ [messages]="messagesSignal()"
133
+ [isLoading]="isLoadingSignal()"
134
+ [inputClass]="assistantMessageClass"
135
+ (thumbsUp)="handleAssistantThumbsUp($event)"
136
+ (thumbsDown)="handleAssistantThumbsDown($event)"
137
+ (readAloud)="handleAssistantReadAloud($event)"
138
+ (regenerate)="handleAssistantRegenerate($event)">
139
+ </copilot-chat-assistant-message>
140
+ }
141
+ } @else if (message && message.role === 'user') {
142
+ <!-- User message with slot support -->
143
+ @if (userMessageComponent || userMessageTemplate) {
144
+ <copilot-slot
145
+ [slot]="userMessageTemplate || userMessageComponent"
146
+ [context]="mergeUserProps(message)"
147
+ [defaultComponent]="defaultUserComponent">
148
+ </copilot-slot>
149
+ } @else {
150
+ <copilot-chat-user-message
151
+ [message]="message"
152
+ [inputClass]="userMessageClass">
153
+ </copilot-chat-user-message>
154
+ }
155
+ }
156
+ }
157
+
158
+ <!-- Cursor - exactly like React's conditional rendering -->
159
+ @if (showCursor) {
160
+ @if (cursorComponent || cursorTemplate) {
161
+ <copilot-slot
162
+ [slot]="cursorTemplate || cursorComponent"
163
+ [context]="{ inputClass: cursorClass }"
164
+ [defaultComponent]="defaultCursorComponent">
165
+ </copilot-slot>
166
+ } @else {
167
+ <copilot-chat-message-view-cursor
168
+ [inputClass]="cursorClass">
169
+ </copilot-chat-message-view-cursor>
170
+ }
171
+ }
172
+ </div>
173
+ }
174
+ `, isInline: true, dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "component", type: CopilotSlotComponent, selector: "copilot-slot", inputs: ["slot", "context", "defaultComponent", "outputs"] }, { kind: "component", type: CopilotChatAssistantMessageComponent, selector: "copilot-chat-assistant-message", inputs: ["markdownRendererClass", "toolbarClass", "copyButtonClass", "thumbsUpButtonClass", "thumbsDownButtonClass", "readAloudButtonClass", "regenerateButtonClass", "toolCallsViewClass", "markdownRendererComponent", "toolbarComponent", "copyButtonComponent", "thumbsUpButtonComponent", "thumbsDownButtonComponent", "readAloudButtonComponent", "regenerateButtonComponent", "toolCallsViewComponent", "message", "messages", "isLoading", "additionalToolbarItems", "toolbarVisible", "inputClass"], outputs: ["thumbsUp", "thumbsDown", "readAloud", "regenerate"] }, { kind: "component", type: CopilotChatUserMessageComponent, selector: "copilot-chat-user-message", inputs: ["messageRendererClass", "toolbarClass", "copyButtonClass", "editButtonClass", "branchNavigationClass", "messageRendererComponent", "toolbarComponent", "copyButtonComponent", "editButtonComponent", "branchNavigationComponent", "message", "branchIndex", "numberOfBranches", "additionalToolbarItems", "inputClass"], outputs: ["editMessage", "switchToBranch"] }, { kind: "component", type: CopilotChatMessageViewCursorComponent, selector: "copilot-chat-message-view-cursor", inputs: ["inputClass"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None });
175
+ }
176
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CopilotChatMessageViewComponent, decorators: [{
177
+ type: Component,
178
+ args: [{
179
+ selector: 'copilot-chat-message-view',
180
+ standalone: true,
181
+ imports: [
182
+ CommonModule,
183
+ CopilotSlotComponent,
184
+ CopilotChatAssistantMessageComponent,
185
+ CopilotChatUserMessageComponent,
186
+ CopilotChatMessageViewCursorComponent
187
+ ],
188
+ changeDetection: ChangeDetectionStrategy.OnPush,
189
+ encapsulation: ViewEncapsulation.None,
190
+ template: `
191
+ <!-- Custom layout template support (render prop pattern) -->
192
+ @if (customLayoutTemplate) {
193
+ <ng-container *ngTemplateOutlet="customLayoutTemplate; context: layoutContext()"></ng-container>
194
+ } @else {
195
+ <!-- Default layout - exact React DOM structure: div with "flex flex-col" classes -->
196
+ <div [class]="computedClass()">
197
+ <!-- Message iteration - simplified without tool calls -->
198
+ @for (message of messagesSignal(); track trackByMessageId($index, message)) {
199
+ @if (message && message.role === 'assistant') {
200
+ <!-- Assistant message with slot support -->
201
+ @if (assistantMessageComponent || assistantMessageTemplate) {
202
+ <copilot-slot
203
+ [slot]="assistantMessageTemplate || assistantMessageComponent"
204
+ [context]="mergeAssistantProps(message)"
205
+ [defaultComponent]="defaultAssistantComponent">
206
+ </copilot-slot>
207
+ } @else {
208
+ <copilot-chat-assistant-message
209
+ [message]="message"
210
+ [messages]="messagesSignal()"
211
+ [isLoading]="isLoadingSignal()"
212
+ [inputClass]="assistantMessageClass"
213
+ (thumbsUp)="handleAssistantThumbsUp($event)"
214
+ (thumbsDown)="handleAssistantThumbsDown($event)"
215
+ (readAloud)="handleAssistantReadAloud($event)"
216
+ (regenerate)="handleAssistantRegenerate($event)">
217
+ </copilot-chat-assistant-message>
218
+ }
219
+ } @else if (message && message.role === 'user') {
220
+ <!-- User message with slot support -->
221
+ @if (userMessageComponent || userMessageTemplate) {
222
+ <copilot-slot
223
+ [slot]="userMessageTemplate || userMessageComponent"
224
+ [context]="mergeUserProps(message)"
225
+ [defaultComponent]="defaultUserComponent">
226
+ </copilot-slot>
227
+ } @else {
228
+ <copilot-chat-user-message
229
+ [message]="message"
230
+ [inputClass]="userMessageClass">
231
+ </copilot-chat-user-message>
232
+ }
233
+ }
234
+ }
235
+
236
+ <!-- Cursor - exactly like React's conditional rendering -->
237
+ @if (showCursor) {
238
+ @if (cursorComponent || cursorTemplate) {
239
+ <copilot-slot
240
+ [slot]="cursorTemplate || cursorComponent"
241
+ [context]="{ inputClass: cursorClass }"
242
+ [defaultComponent]="defaultCursorComponent">
243
+ </copilot-slot>
244
+ } @else {
245
+ <copilot-chat-message-view-cursor
246
+ [inputClass]="cursorClass">
247
+ </copilot-chat-message-view-cursor>
248
+ }
249
+ }
250
+ </div>
251
+ }
252
+ `
253
+ }]
254
+ }], propDecorators: { messages: [{
255
+ type: Input
256
+ }], showCursor: [{
257
+ type: Input
258
+ }], isLoading: [{
259
+ type: Input
260
+ }], inputClass: [{
261
+ type: Input
262
+ }], assistantMessageComponent: [{
263
+ type: Input
264
+ }], assistantMessageTemplate: [{
265
+ type: Input
266
+ }], assistantMessageClass: [{
267
+ type: Input
268
+ }], userMessageComponent: [{
269
+ type: Input
270
+ }], userMessageTemplate: [{
271
+ type: Input
272
+ }], userMessageClass: [{
273
+ type: Input
274
+ }], cursorComponent: [{
275
+ type: Input
276
+ }], cursorTemplate: [{
277
+ type: Input
278
+ }], cursorClass: [{
279
+ type: Input
280
+ }], customLayoutTemplate: [{
281
+ type: ContentChild,
282
+ args: ['customLayout']
283
+ }], assistantMessageThumbsUp: [{
284
+ type: Output
285
+ }], assistantMessageThumbsDown: [{
286
+ type: Output
287
+ }], assistantMessageReadAloud: [{
288
+ type: Output
289
+ }], assistantMessageRegenerate: [{
290
+ type: Output
291
+ }], userMessageCopy: [{
292
+ type: Output
293
+ }], userMessageEdit: [{
294
+ type: Output
295
+ }] } });
296
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"copilot-chat-message-view.component.js","sourceRoot":"","sources":["../../../../src/components/chat/copilot-chat-message-view.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  Input,\n  Output,\n  EventEmitter,\n  ContentChild,\n  TemplateRef,\n  Type,\n  ChangeDetectionStrategy,\n  ViewEncapsulation,\n  signal,\n  computed,\n  OnInit,\n  OnChanges\n} from '@angular/core';\nimport { CommonModule } from '@angular/common';\nimport { CopilotSlotComponent } from '../../lib/slots/copilot-slot.component';\nimport type { Message } from '@ag-ui/core';\nimport { CopilotChatAssistantMessageComponent } from './copilot-chat-assistant-message.component';\nimport { CopilotChatUserMessageComponent } from './copilot-chat-user-message.component';\nimport { CopilotChatMessageViewCursorComponent } from './copilot-chat-message-view-cursor.component';\nimport { cn } from '../../lib/utils';\n\n/**\n * CopilotChatMessageView component - Angular port of the React component.\n * Renders a list of chat messages with support for custom slots and layouts.\n * DOM structure and Tailwind classes match the React implementation exactly.\n */\n@Component({\n  selector: 'copilot-chat-message-view',\n  standalone: true,\n  imports: [\n    CommonModule,\n    CopilotSlotComponent,\n    CopilotChatAssistantMessageComponent,\n    CopilotChatUserMessageComponent,\n    CopilotChatMessageViewCursorComponent\n  ],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  encapsulation: ViewEncapsulation.None,\n  template: `\n    <!-- Custom layout template support (render prop pattern) -->\n    @if (customLayoutTemplate) {\n      <ng-container *ngTemplateOutlet=\"customLayoutTemplate; context: layoutContext()\"></ng-container>\n    } @else {\n      <!-- Default layout - exact React DOM structure: div with \"flex flex-col\" classes -->\n      <div [class]=\"computedClass()\">\n        <!-- Message iteration - simplified without tool calls -->\n        @for (message of messagesSignal(); track trackByMessageId($index, message)) {\n          @if (message && message.role === 'assistant') {\n            <!-- Assistant message with slot support -->\n            @if (assistantMessageComponent || assistantMessageTemplate) {\n              <copilot-slot\n                [slot]=\"assistantMessageTemplate || assistantMessageComponent\"\n                [context]=\"mergeAssistantProps(message)\"\n                [defaultComponent]=\"defaultAssistantComponent\">\n              </copilot-slot>\n            } @else {\n              <copilot-chat-assistant-message \n                [message]=\"message\"\n                [messages]=\"messagesSignal()\"\n                [isLoading]=\"isLoadingSignal()\"\n                [inputClass]=\"assistantMessageClass\"\n                (thumbsUp)=\"handleAssistantThumbsUp($event)\"\n                (thumbsDown)=\"handleAssistantThumbsDown($event)\"\n                (readAloud)=\"handleAssistantReadAloud($event)\"\n                (regenerate)=\"handleAssistantRegenerate($event)\">\n              </copilot-chat-assistant-message>\n            }\n          } @else if (message && message.role === 'user') {\n            <!-- User message with slot support -->\n            @if (userMessageComponent || userMessageTemplate) {\n              <copilot-slot\n                [slot]=\"userMessageTemplate || userMessageComponent\"\n                [context]=\"mergeUserProps(message)\"\n                [defaultComponent]=\"defaultUserComponent\">\n              </copilot-slot>\n            } @else {\n              <copilot-chat-user-message \n                [message]=\"message\"\n                [inputClass]=\"userMessageClass\">\n              </copilot-chat-user-message>\n            }\n          }\n        }\n        \n        <!-- Cursor - exactly like React's conditional rendering -->\n        @if (showCursor) {\n          @if (cursorComponent || cursorTemplate) {\n            <copilot-slot\n              [slot]=\"cursorTemplate || cursorComponent\"\n              [context]=\"{ inputClass: cursorClass }\"\n              [defaultComponent]=\"defaultCursorComponent\">\n            </copilot-slot>\n          } @else {\n            <copilot-chat-message-view-cursor \n              [inputClass]=\"cursorClass\">\n            </copilot-chat-message-view-cursor>\n          }\n        }\n      </div>\n    }\n  `\n})\nexport class CopilotChatMessageViewComponent implements OnInit, OnChanges {\n  // Core inputs matching React props\n  @Input() messages: Message[] = [];\n  @Input() showCursor = false;\n  @Input() isLoading = false;\n  @Input() inputClass?: string;\n  \n  // Handler availability handled via DI service\n  \n  // Assistant message slot inputs\n  @Input() assistantMessageComponent?: Type<any>;\n  @Input() assistantMessageTemplate?: TemplateRef<any>;\n  @Input() assistantMessageClass?: string;\n  \n  // User message slot inputs\n  @Input() userMessageComponent?: Type<any>;\n  @Input() userMessageTemplate?: TemplateRef<any>;\n  @Input() userMessageClass?: string;\n  \n  \n  // Cursor slot inputs\n  @Input() cursorComponent?: Type<any>;\n  @Input() cursorTemplate?: TemplateRef<any>;\n  @Input() cursorClass?: string;\n  \n  // Custom layout template (render prop pattern)\n  @ContentChild('customLayout') customLayoutTemplate?: TemplateRef<any>;\n  \n  // Output events (bubbled from child components)\n  @Output() assistantMessageThumbsUp = new EventEmitter<{ message: Message }>();\n  @Output() assistantMessageThumbsDown = new EventEmitter<{ message: Message }>();\n  @Output() assistantMessageReadAloud = new EventEmitter<{ message: Message }>();\n  @Output() assistantMessageRegenerate = new EventEmitter<{ message: Message }>();\n  @Output() userMessageCopy = new EventEmitter<{ message: Message }>();\n  @Output() userMessageEdit = new EventEmitter<{ message: Message }>();\n  \n  // Default components for slots\n  protected readonly defaultAssistantComponent = CopilotChatAssistantMessageComponent;\n  protected readonly defaultUserComponent = CopilotChatUserMessageComponent;\n  protected readonly defaultCursorComponent = CopilotChatMessageViewCursorComponent;\n  \n  // Signals for reactive updates\n  protected messagesSignal = signal<Message[]>([]);\n  protected showCursorSignal = signal(false);\n  protected isLoadingSignal = signal(false);\n  protected inputClassSignal = signal<string | undefined>(undefined);\n  \n  // Computed class matching React: twMerge(\"flex flex-col\", className)\n  computedClass = computed(() => \n    cn('flex flex-col', this.inputClassSignal())\n  );\n  \n  \n  // Layout context for custom templates (render prop pattern)\n  layoutContext = computed(() => ({\n    isLoading: this.isLoadingSignal(),\n    messages: this.messagesSignal(),\n    showCursor: this.showCursorSignal(),\n    messageElements: this.messagesSignal().filter(m => m && (m.role === 'assistant' || m.role === 'user'))\n  }));\n  \n  // Slot resolution computed signals\n  assistantMessageSlot = computed(() => \n    this.assistantMessageComponent || this.assistantMessageClass\n  );\n  \n  userMessageSlot = computed(() => \n    this.userMessageComponent || this.userMessageClass\n  );\n  \n  cursorSlot = computed(() => \n    this.cursorComponent || this.cursorClass\n  );\n  \n  // Props merging helpers\n  mergeAssistantProps(message: Message) {\n    return {\n      message,\n      messages: this.messagesSignal(),\n      isLoading: this.isLoadingSignal(),\n      inputClass: this.assistantMessageClass\n    };\n  }\n  \n  mergeUserProps(message: Message) {\n    return {\n      message,\n      inputClass: this.userMessageClass\n    };\n  }\n  \n  // TrackBy function for performance optimization\n  trackByMessageId(index: number, message: Message): string {\n    return message?.id || `index-${index}`;\n  }\n  \n  // Lifecycle hooks\n  ngOnInit() {\n    // Initialize signals with input values\n    this.messagesSignal.set(this.messages);\n    this.showCursorSignal.set(this.showCursor);\n    this.isLoadingSignal.set(this.isLoading);\n    this.inputClassSignal.set(this.inputClass);\n  }\n  \n  ngOnChanges() {\n    this.messagesSignal.set(this.messages);\n    this.showCursorSignal.set(this.showCursor);\n    this.isLoadingSignal.set(this.isLoading);\n    this.inputClassSignal.set(this.inputClass);\n  }\n  \n  // Event handlers - just pass them through\n  handleAssistantThumbsUp(event: { message: Message }): void {\n    this.assistantMessageThumbsUp.emit(event);\n  }\n  \n  handleAssistantThumbsDown(event: { message: Message }): void {\n    this.assistantMessageThumbsDown.emit(event);\n  }\n  \n  handleAssistantReadAloud(event: { message: Message }): void {\n    this.assistantMessageReadAloud.emit(event);\n  }\n  \n  handleAssistantRegenerate(event: { message: Message }): void {\n    this.assistantMessageRegenerate.emit(event);\n  }\n}\n"]}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29waWxvdC1jaGF0LW1lc3NhZ2Utdmlldy50eXBlcy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3NyYy9jb21wb25lbnRzL2NoYXQvY29waWxvdC1jaGF0LW1lc3NhZ2Utdmlldy50eXBlcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgTWVzc2FnZSB9IGZyb20gJ0BhZy11aS9jbGllbnQnO1xuaW1wb3J0IHsgVHlwZSwgVGVtcGxhdGVSZWYgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcblxuLy8gQ29udGV4dCBpbnRlcmZhY2VzIGZvciB0ZW1wbGF0ZSBzbG90c1xuZXhwb3J0IGludGVyZmFjZSBNZXNzYWdlVmlld0NvbnRleHQge1xuICBzaG93Q3Vyc29yOiBib29sZWFuO1xuICBtZXNzYWdlczogTWVzc2FnZVtdO1xuICBtZXNzYWdlRWxlbWVudHM6IGFueVtdOyAvLyBXaWxsIGJlIHBvcHVsYXRlZCB3aXRoIHJlbmRlcmVkIGVsZW1lbnRzXG59XG5cbi8vIGVzbGludC1kaXNhYmxlLW5leHQtbGluZSBAdHlwZXNjcmlwdC1lc2xpbnQvbm8tZW1wdHktb2JqZWN0LXR5cGVcbmV4cG9ydCBpbnRlcmZhY2UgQ3Vyc29yQ29udGV4dCB7XG4gIC8vIEVtcHR5IGZvciBub3csIGNhbiBiZSBleHRlbmRlZCBpZiBuZWVkZWRcbn1cblxuLy8gQ29tcG9uZW50IGlucHV0IHByb3BzIGludGVyZmFjZVxuZXhwb3J0IGludGVyZmFjZSBDb3BpbG90Q2hhdE1lc3NhZ2VWaWV3UHJvcHMge1xuICBtZXNzYWdlcz86IE1lc3NhZ2VbXTtcbiAgc2hvd0N1cnNvcj86IGJvb2xlYW47XG4gIGlucHV0Q2xhc3M/OiBzdHJpbmc7XG4gIFxuICAvLyBBc3Npc3RhbnQgbWVzc2FnZSBzbG90c1xuICBhc3Npc3RhbnRNZXNzYWdlQ29tcG9uZW50PzogVHlwZTxhbnk+O1xuICBhc3Npc3RhbnRNZXNzYWdlVGVtcGxhdGU/OiBUZW1wbGF0ZVJlZjxhbnk+O1xuICBhc3Npc3RhbnRNZXNzYWdlQ2xhc3M/OiBzdHJpbmc7XG4gIFxuICAvLyBVc2VyIG1lc3NhZ2Ugc2xvdHNcbiAgdXNlck1lc3NhZ2VDb21wb25lbnQ/OiBUeXBlPGFueT47XG4gIHVzZXJNZXNzYWdlVGVtcGxhdGU/OiBUZW1wbGF0ZVJlZjxhbnk+O1xuICB1c2VyTWVzc2FnZUNsYXNzPzogc3RyaW5nO1xuICBcbiAgLy8gQ3Vyc29yIHNsb3RzXG4gIGN1cnNvckNvbXBvbmVudD86IFR5cGU8YW55PjtcbiAgY3Vyc29yVGVtcGxhdGU/OiBUZW1wbGF0ZVJlZjxhbnk+O1xuICBjdXJzb3JDbGFzcz86IHN0cmluZztcbn1cblxuLy8gUmUtZXhwb3J0IGZvciBjb252ZW5pZW5jZVxuZXhwb3J0IHR5cGUgeyBNZXNzYWdlIH07Il19
@@ -0,0 +1,188 @@
1
+ import { Component, Input, Output, EventEmitter, ElementRef, signal, computed, effect, inject, ChangeDetectionStrategy, ViewEncapsulation } from '@angular/core';
2
+ import { CopilotChatConfigurationService } from '../../core/chat-configuration/chat-configuration.service';
3
+ import { cn } from '../../lib/utils';
4
+ import * as i0 from "@angular/core";
5
+ export class CopilotChatTextareaComponent {
6
+ elementRef = inject((ElementRef));
7
+ get textareaRef() { return this.elementRef; }
8
+ set inputValue(val) {
9
+ this.value.set(val || '');
10
+ }
11
+ set inputPlaceholder(val) {
12
+ this.customPlaceholder.set(val);
13
+ }
14
+ set inputMaxRows(val) {
15
+ this.maxRows.set(val || 5);
16
+ }
17
+ set inputAutoFocus(val) {
18
+ this.autoFocus.set(val ?? true);
19
+ }
20
+ set inputDisabled(val) {
21
+ this.disabled.set(val || false);
22
+ }
23
+ set inputClass(val) {
24
+ this.customClass.set(val);
25
+ }
26
+ valueChange = new EventEmitter();
27
+ keyDown = new EventEmitter();
28
+ chatConfig = inject(CopilotChatConfigurationService);
29
+ // Signals for reactive state
30
+ value = signal('');
31
+ customPlaceholder = signal(undefined);
32
+ maxRows = signal(5);
33
+ autoFocus = signal(true);
34
+ disabled = signal(false);
35
+ customClass = signal(undefined);
36
+ maxHeight = signal(0);
37
+ // Computed values
38
+ placeholder = computed(() => {
39
+ return this.customPlaceholder() || this.chatConfig.labels().chatInputPlaceholder;
40
+ });
41
+ computedClass = computed(() => {
42
+ const baseClasses = cn(
43
+ // Layout and sizing
44
+ 'w-full p-5 pb-0',
45
+ // Behavior
46
+ 'outline-none resize-none',
47
+ // Background
48
+ 'bg-transparent',
49
+ // Typography
50
+ 'antialiased font-regular leading-relaxed text-[16px]',
51
+ // Placeholder styles
52
+ 'placeholder:text-[#00000077] dark:placeholder:text-[#fffc]');
53
+ return cn(baseClasses, this.customClass());
54
+ });
55
+ constructor() {
56
+ // Effect to sync value with chat configuration if available
57
+ effect(() => {
58
+ const configValue = this.chatConfig.inputValue();
59
+ if (configValue !== undefined && !this.customPlaceholder()) {
60
+ this.value.set(configValue);
61
+ }
62
+ }, { allowSignalWrites: true });
63
+ }
64
+ ngAfterViewInit() {
65
+ this.calculateMaxHeight();
66
+ this.adjustHeight();
67
+ if (this.autoFocus()) {
68
+ setTimeout(() => {
69
+ this.elementRef.nativeElement.focus();
70
+ });
71
+ }
72
+ }
73
+ ngOnChanges(changes) {
74
+ if (changes['inputMaxRows']) {
75
+ this.calculateMaxHeight();
76
+ }
77
+ }
78
+ onInput(event) {
79
+ const textarea = event.target;
80
+ const newValue = textarea.value;
81
+ this.value.set(newValue);
82
+ this.valueChange.emit(newValue);
83
+ // Update chat configuration if available
84
+ if (this.chatConfig) {
85
+ this.chatConfig.setInputValue(newValue);
86
+ }
87
+ this.adjustHeight();
88
+ }
89
+ onKeyDown(event) {
90
+ // Check for Enter key without Shift
91
+ if (event.key === 'Enter' && !event.shiftKey) {
92
+ event.preventDefault();
93
+ this.keyDown.emit(event);
94
+ }
95
+ else {
96
+ this.keyDown.emit(event);
97
+ }
98
+ }
99
+ calculateMaxHeight() {
100
+ const textarea = this.elementRef.nativeElement;
101
+ const maxRowsValue = this.maxRows();
102
+ // Save current value
103
+ const currentValue = textarea.value;
104
+ // Clear content to measure single row height
105
+ textarea.value = '';
106
+ textarea.style.height = 'auto';
107
+ // Get computed styles to account for padding
108
+ const computedStyle = window.getComputedStyle(textarea);
109
+ const paddingTop = parseFloat(computedStyle.paddingTop);
110
+ const paddingBottom = parseFloat(computedStyle.paddingBottom);
111
+ // Calculate actual content height (without padding)
112
+ const contentHeight = textarea.scrollHeight - paddingTop - paddingBottom;
113
+ // Calculate max height: content height for maxRows + padding
114
+ const calculatedMaxHeight = contentHeight * maxRowsValue + paddingTop + paddingBottom;
115
+ this.maxHeight.set(calculatedMaxHeight);
116
+ // Restore original value
117
+ textarea.value = currentValue;
118
+ // Adjust height after calculating maxHeight
119
+ if (currentValue) {
120
+ this.adjustHeight();
121
+ }
122
+ }
123
+ adjustHeight() {
124
+ const textarea = this.elementRef.nativeElement;
125
+ const maxHeightValue = this.maxHeight();
126
+ if (maxHeightValue > 0) {
127
+ textarea.style.height = 'auto';
128
+ textarea.style.height = `${Math.min(textarea.scrollHeight, maxHeightValue)}px`;
129
+ }
130
+ }
131
+ /**
132
+ * Public method to focus the textarea
133
+ */
134
+ focus() {
135
+ this.elementRef.nativeElement.focus();
136
+ }
137
+ /**
138
+ * Public method to get current value
139
+ */
140
+ getValue() {
141
+ return this.value();
142
+ }
143
+ /**
144
+ * Public method to set value programmatically
145
+ */
146
+ setValue(value) {
147
+ this.value.set(value);
148
+ this.valueChange.emit(value);
149
+ if (this.chatConfig) {
150
+ this.chatConfig.setInputValue(value);
151
+ }
152
+ setTimeout(() => this.adjustHeight());
153
+ }
154
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CopilotChatTextareaComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
155
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: CopilotChatTextareaComponent, isStandalone: true, selector: "textarea[copilotChatTextarea]", inputs: { inputValue: "inputValue", inputPlaceholder: "inputPlaceholder", inputMaxRows: "inputMaxRows", inputAutoFocus: "inputAutoFocus", inputDisabled: "inputDisabled", inputClass: "inputClass" }, outputs: { valueChange: "valueChange", keyDown: "keyDown" }, host: { listeners: { "input": "onInput($event)", "keydown": "onKeyDown($event)" }, properties: { "value": "value()", "placeholder": "placeholder()", "disabled": "disabled()", "class": "computedClass()", "style.max-height.px": "maxHeight()", "style.overflow": "'auto'", "style.resize": "'none'", "attr.rows": "1" } }, usesOnChanges: true, ngImport: i0, template: '', isInline: true, changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None });
156
+ }
157
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CopilotChatTextareaComponent, decorators: [{
158
+ type: Component,
159
+ args: [{ selector: 'textarea[copilotChatTextarea]', standalone: true, imports: [], changeDetection: ChangeDetectionStrategy.OnPush, encapsulation: ViewEncapsulation.None, host: {
160
+ '[value]': 'value()',
161
+ '[placeholder]': 'placeholder()',
162
+ '[disabled]': 'disabled()',
163
+ '[class]': 'computedClass()',
164
+ '[style.max-height.px]': 'maxHeight()',
165
+ '[style.overflow]': "'auto'",
166
+ '[style.resize]': "'none'",
167
+ '(input)': 'onInput($event)',
168
+ '(keydown)': 'onKeyDown($event)',
169
+ '[attr.rows]': '1'
170
+ }, template: '' }]
171
+ }], ctorParameters: () => [], propDecorators: { inputValue: [{
172
+ type: Input
173
+ }], inputPlaceholder: [{
174
+ type: Input
175
+ }], inputMaxRows: [{
176
+ type: Input
177
+ }], inputAutoFocus: [{
178
+ type: Input
179
+ }], inputDisabled: [{
180
+ type: Input
181
+ }], inputClass: [{
182
+ type: Input
183
+ }], valueChange: [{
184
+ type: Output
185
+ }], keyDown: [{
186
+ type: Output
187
+ }] } });
188
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"copilot-chat-textarea.component.js","sourceRoot":"","sources":["../../../../src/components/chat/copilot-chat-textarea.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  Input,\n  Output,\n  EventEmitter,\n  ElementRef,\n  AfterViewInit,\n  OnChanges,\n  SimpleChanges,\n  signal,\n  computed,\n  effect,\n  inject,\n  ChangeDetectionStrategy,\n  ViewEncapsulation\n} from '@angular/core';\nimport { CopilotChatConfigurationService } from '../../core/chat-configuration/chat-configuration.service';\nimport { cn } from '../../lib/utils';\n\n@Component({\n  selector: 'textarea[copilotChatTextarea]',\n  standalone: true,\n  imports: [],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  encapsulation: ViewEncapsulation.None,\n  host: {\n    '[value]': 'value()',\n    '[placeholder]': 'placeholder()',\n    '[disabled]': 'disabled()',\n    '[class]': 'computedClass()',\n    '[style.max-height.px]': 'maxHeight()',\n    '[style.overflow]': \"'auto'\",\n    '[style.resize]': \"'none'\",\n    '(input)': 'onInput($event)',\n    '(keydown)': 'onKeyDown($event)',\n    '[attr.rows]': '1'\n  },\n  template: '',\n  styles: []\n})\nexport class CopilotChatTextareaComponent implements AfterViewInit, OnChanges {\n  private elementRef = inject(ElementRef<HTMLTextAreaElement>);\n  get textareaRef() { return this.elementRef; }\n  \n  @Input() set inputValue(val: string | undefined) {\n    this.value.set(val || '');\n  }\n  @Input() set inputPlaceholder(val: string | undefined) {\n    this.customPlaceholder.set(val);\n  }\n  @Input() set inputMaxRows(val: number | undefined) {\n    this.maxRows.set(val || 5);\n  }\n  @Input() set inputAutoFocus(val: boolean | undefined) {\n    this.autoFocus.set(val ?? true);\n  }\n  @Input() set inputDisabled(val: boolean | undefined) {\n    this.disabled.set(val || false);\n  }\n  @Input() set inputClass(val: string | undefined) {\n    this.customClass.set(val);\n  }\n  \n  @Output() valueChange = new EventEmitter<string>();\n  @Output() keyDown = new EventEmitter<KeyboardEvent>();\n  \n  private chatConfig = inject(CopilotChatConfigurationService);\n  \n  // Signals for reactive state\n  value = signal<string>('');\n  customPlaceholder = signal<string | undefined>(undefined);\n  maxRows = signal<number>(5);\n  autoFocus = signal<boolean>(true);\n  disabled = signal<boolean>(false);\n  customClass = signal<string | undefined>(undefined);\n  maxHeight = signal<number>(0);\n  \n  // Computed values\n  placeholder = computed(() => {\n    return this.customPlaceholder() || this.chatConfig.labels().chatInputPlaceholder;\n  });\n  \n  computedClass = computed(() => {\n    const baseClasses = cn(\n      // Layout and sizing\n      'w-full p-5 pb-0',\n      // Behavior\n      'outline-none resize-none',\n      // Background\n      'bg-transparent',\n      // Typography\n      'antialiased font-regular leading-relaxed text-[16px]',\n      // Placeholder styles\n      'placeholder:text-[#00000077] dark:placeholder:text-[#fffc]'\n    );\n    return cn(baseClasses, this.customClass());\n  });\n  \n  constructor() {\n    // Effect to sync value with chat configuration if available\n    effect(\n      () => {\n        const configValue = this.chatConfig.inputValue();\n        if (configValue !== undefined && !this.customPlaceholder()) {\n          this.value.set(configValue);\n        }\n      },\n      { allowSignalWrites: true }\n    );\n  }\n  \n  ngAfterViewInit(): void {\n    this.calculateMaxHeight();\n    this.adjustHeight();\n    \n    if (this.autoFocus()) {\n      setTimeout(() => {\n        this.elementRef.nativeElement.focus();\n      });\n    }\n  }\n  \n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes['inputMaxRows']) {\n      this.calculateMaxHeight();\n    }\n  }\n  \n  onInput(event: Event): void {\n    const textarea = event.target as HTMLTextAreaElement;\n    const newValue = textarea.value;\n    \n    this.value.set(newValue);\n    this.valueChange.emit(newValue);\n    \n    // Update chat configuration if available\n    if (this.chatConfig) {\n      this.chatConfig.setInputValue(newValue);\n    }\n    \n    this.adjustHeight();\n  }\n  \n  onKeyDown(event: KeyboardEvent): void {\n    // Check for Enter key without Shift\n    if (event.key === 'Enter' && !event.shiftKey) {\n      event.preventDefault();\n      this.keyDown.emit(event);\n    } else {\n      this.keyDown.emit(event);\n    }\n  }\n  \n  private calculateMaxHeight(): void {\n    const textarea = this.elementRef.nativeElement;\n    const maxRowsValue = this.maxRows();\n    \n    // Save current value\n    const currentValue = textarea.value;\n    \n    // Clear content to measure single row height\n    textarea.value = '';\n    textarea.style.height = 'auto';\n    \n    // Get computed styles to account for padding\n    const computedStyle = window.getComputedStyle(textarea);\n    const paddingTop = parseFloat(computedStyle.paddingTop);\n    const paddingBottom = parseFloat(computedStyle.paddingBottom);\n    \n    // Calculate actual content height (without padding)\n    const contentHeight = textarea.scrollHeight - paddingTop - paddingBottom;\n    \n    // Calculate max height: content height for maxRows + padding\n    const calculatedMaxHeight = contentHeight * maxRowsValue + paddingTop + paddingBottom;\n    this.maxHeight.set(calculatedMaxHeight);\n    \n    // Restore original value\n    textarea.value = currentValue;\n    \n    // Adjust height after calculating maxHeight\n    if (currentValue) {\n      this.adjustHeight();\n    }\n  }\n  \n  private adjustHeight(): void {\n    const textarea = this.elementRef.nativeElement;\n    const maxHeightValue = this.maxHeight();\n    \n    if (maxHeightValue > 0) {\n      textarea.style.height = 'auto';\n      textarea.style.height = `${Math.min(textarea.scrollHeight, maxHeightValue)}px`;\n    }\n  }\n  \n  /**\n   * Public method to focus the textarea\n   */\n  focus(): void {\n    this.elementRef.nativeElement.focus();\n  }\n  \n  /**\n   * Public method to get current value\n   */\n  getValue(): string {\n    return this.value();\n  }\n  \n  /**\n   * Public method to set value programmatically\n   */\n  setValue(value: string): void {\n    this.value.set(value);\n    this.valueChange.emit(value);\n    \n    if (this.chatConfig) {\n      this.chatConfig.setInputValue(value);\n    }\n    \n    setTimeout(() => this.adjustHeight());\n  }\n}\n"]}