@copilotkitnext/angular 0.0.1 → 0.0.4

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (171) hide show
  1. package/README.md +248 -0
  2. package/dist/README.md +248 -0
  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/esm2022/components/chat/copilot-chat-assistant-message-buttons.component.mjs +384 -0
  7. package/dist/esm2022/components/chat/copilot-chat-assistant-message-renderer.component.mjs +286 -0
  8. package/dist/esm2022/components/chat/copilot-chat-assistant-message-toolbar.component.mjs +27 -0
  9. package/dist/esm2022/components/chat/copilot-chat-assistant-message.component.mjs +433 -0
  10. package/dist/esm2022/components/chat/copilot-chat-assistant-message.types.mjs +2 -0
  11. package/dist/esm2022/components/chat/copilot-chat-audio-recorder.component.mjs +202 -0
  12. package/dist/esm2022/components/chat/copilot-chat-buttons.component.mjs +321 -0
  13. package/dist/esm2022/components/chat/copilot-chat-input-defaults.mjs +38 -0
  14. package/dist/esm2022/components/chat/copilot-chat-input.component.mjs +666 -0
  15. package/dist/esm2022/components/chat/copilot-chat-input.types.mjs +10 -0
  16. package/dist/esm2022/components/chat/copilot-chat-message-view-cursor.component.mjs +45 -0
  17. package/dist/esm2022/components/chat/copilot-chat-message-view.component.mjs +296 -0
  18. package/dist/esm2022/components/chat/copilot-chat-message-view.types.mjs +2 -0
  19. package/dist/esm2022/components/chat/copilot-chat-textarea.component.mjs +188 -0
  20. package/dist/esm2022/components/chat/copilot-chat-tool-calls-view.component.mjs +216 -0
  21. package/dist/esm2022/components/chat/copilot-chat-toolbar.component.mjs +25 -0
  22. package/dist/esm2022/components/chat/copilot-chat-tools-menu.component.mjs +199 -0
  23. package/dist/esm2022/components/chat/copilot-chat-user-message-branch-navigation.component.mjs +137 -0
  24. package/dist/esm2022/components/chat/copilot-chat-user-message-buttons.component.mjs +207 -0
  25. package/dist/esm2022/components/chat/copilot-chat-user-message-renderer.component.mjs +35 -0
  26. package/dist/esm2022/components/chat/copilot-chat-user-message-toolbar.component.mjs +34 -0
  27. package/dist/esm2022/components/chat/copilot-chat-user-message.component.mjs +341 -0
  28. package/dist/esm2022/components/chat/copilot-chat-user-message.types.mjs +2 -0
  29. package/dist/esm2022/components/chat/copilot-chat-view-disclaimer.component.mjs +52 -0
  30. package/dist/esm2022/components/chat/copilot-chat-view-feather.component.mjs +55 -0
  31. package/dist/esm2022/components/chat/copilot-chat-view-handlers.service.mjs +19 -0
  32. package/dist/esm2022/components/chat/copilot-chat-view-input-container.component.mjs +110 -0
  33. package/dist/esm2022/components/chat/copilot-chat-view-scroll-to-bottom-button.component.mjs +93 -0
  34. package/dist/esm2022/components/chat/copilot-chat-view-scroll-view.component.mjs +443 -0
  35. package/dist/esm2022/components/chat/copilot-chat-view.component.mjs +479 -0
  36. package/dist/esm2022/components/chat/copilot-chat-view.types.mjs +2 -0
  37. package/dist/esm2022/components/chat/copilot-chat.component.mjs +214 -0
  38. package/dist/esm2022/components/copilotkit-tool-render.component.mjs +153 -0
  39. package/dist/esm2022/copilotkitnext-angular.mjs +5 -0
  40. package/dist/esm2022/core/chat-configuration/chat-configuration.providers.mjs +65 -0
  41. package/dist/esm2022/core/chat-configuration/chat-configuration.service.mjs +145 -0
  42. package/dist/esm2022/core/chat-configuration/chat-configuration.types.mjs +26 -0
  43. package/dist/esm2022/core/copilotkit.providers.mjs +34 -0
  44. package/dist/esm2022/core/copilotkit.service.mjs +430 -0
  45. package/dist/esm2022/core/copilotkit.types.mjs +12 -0
  46. package/dist/esm2022/directives/copilotkit-agent-context.directive.mjs +130 -0
  47. package/dist/esm2022/directives/copilotkit-agent.directive.mjs +217 -0
  48. package/dist/esm2022/directives/copilotkit-chat-config.directive.mjs +218 -0
  49. package/dist/esm2022/directives/copilotkit-config.directive.mjs +94 -0
  50. package/dist/esm2022/directives/copilotkit-frontend-tool.directive.mjs +130 -0
  51. package/dist/esm2022/directives/copilotkit-human-in-the-loop.directive.mjs +266 -0
  52. package/dist/esm2022/directives/stick-to-bottom.directive.mjs +181 -0
  53. package/dist/esm2022/index.mjs +70 -0
  54. package/dist/esm2022/lib/directives/tooltip.directive.mjs +211 -0
  55. package/dist/esm2022/lib/slots/copilot-slot.component.mjs +144 -0
  56. package/dist/esm2022/lib/slots/slot.types.mjs +6 -0
  57. package/dist/esm2022/lib/slots/slot.utils.mjs +222 -0
  58. package/dist/esm2022/lib/utils.mjs +10 -0
  59. package/dist/esm2022/services/resize-observer.service.mjs +152 -0
  60. package/dist/esm2022/services/scroll-position.service.mjs +124 -0
  61. package/dist/esm2022/types/frontend-tool.mjs +2 -0
  62. package/dist/esm2022/types/human-in-the-loop.mjs +2 -0
  63. package/dist/esm2022/utils/agent-context.utils.mjs +114 -0
  64. package/dist/esm2022/utils/agent.utils.mjs +204 -0
  65. package/dist/esm2022/utils/chat-config.utils.mjs +186 -0
  66. package/dist/esm2022/utils/copilotkit.utils.mjs +20 -0
  67. package/dist/esm2022/utils/frontend-tool.utils.mjs +228 -0
  68. package/dist/esm2022/utils/human-in-the-loop.utils.mjs +296 -0
  69. package/dist/fesm2022/copilotkitnext-angular.mjs +163 -164
  70. package/dist/fesm2022/copilotkitnext-angular.mjs.map +1 -1
  71. package/dist/styles.css +0 -27
  72. package/package.json +23 -20
  73. package/vitest.config.mts +32 -21
  74. package/.turbo/turbo-build.log +0 -39
  75. package/.turbo/turbo-check-types.log +0 -0
  76. package/.turbo/turbo-test.log +0 -71
  77. package/README-agent-context.md +0 -310
  78. package/ng-package.json +0 -19
  79. package/slots.md +0 -331
  80. package/src/components/chat/__tests__/copilot-chat-assistant-message.component.spec.ts +0 -282
  81. package/src/components/chat/__tests__/copilot-chat-input.component.spec.ts +0 -419
  82. package/src/components/chat/__tests__/copilot-chat-message-view.component.spec.ts +0 -372
  83. package/src/components/chat/__tests__/copilot-chat-user-message.component.spec.ts +0 -249
  84. package/src/components/chat/copilot-chat-assistant-message-buttons.component.ts +0 -292
  85. package/src/components/chat/copilot-chat-assistant-message-renderer.component.ts +0 -472
  86. package/src/components/chat/copilot-chat-assistant-message-toolbar.component.ts +0 -29
  87. package/src/components/chat/copilot-chat-assistant-message.component.ts +0 -463
  88. package/src/components/chat/copilot-chat-assistant-message.types.ts +0 -50
  89. package/src/components/chat/copilot-chat-audio-recorder.component.ts +0 -241
  90. package/src/components/chat/copilot-chat-buttons.component.ts +0 -308
  91. package/src/components/chat/copilot-chat-buttons.component.ts.bak +0 -471
  92. package/src/components/chat/copilot-chat-input-defaults.ts +0 -47
  93. package/src/components/chat/copilot-chat-input.component.ts +0 -512
  94. package/src/components/chat/copilot-chat-input.types.ts +0 -148
  95. package/src/components/chat/copilot-chat-message-view-cursor.component.ts +0 -51
  96. package/src/components/chat/copilot-chat-message-view.component.ts +0 -233
  97. package/src/components/chat/copilot-chat-message-view.types.ts +0 -39
  98. package/src/components/chat/copilot-chat-textarea.component.ts +0 -220
  99. package/src/components/chat/copilot-chat-tool-calls-view.component.ts +0 -261
  100. package/src/components/chat/copilot-chat-toolbar.component.ts +0 -35
  101. package/src/components/chat/copilot-chat-tools-menu.component.ts +0 -185
  102. package/src/components/chat/copilot-chat-user-message-branch-navigation.component.ts +0 -121
  103. package/src/components/chat/copilot-chat-user-message-buttons.component.ts +0 -170
  104. package/src/components/chat/copilot-chat-user-message-renderer.component.ts +0 -37
  105. package/src/components/chat/copilot-chat-user-message-toolbar.component.ts +0 -37
  106. package/src/components/chat/copilot-chat-user-message.component.ts +0 -247
  107. package/src/components/chat/copilot-chat-user-message.types.ts +0 -42
  108. package/src/components/chat/copilot-chat-view-disclaimer.component.ts +0 -51
  109. package/src/components/chat/copilot-chat-view-feather.component.ts +0 -47
  110. package/src/components/chat/copilot-chat-view-handlers.service.ts +0 -14
  111. package/src/components/chat/copilot-chat-view-input-container.component.ts +0 -87
  112. package/src/components/chat/copilot-chat-view-scroll-to-bottom-button.component.ts +0 -79
  113. package/src/components/chat/copilot-chat-view-scroll-view.component.ts +0 -322
  114. package/src/components/chat/copilot-chat-view.component.ts +0 -420
  115. package/src/components/chat/copilot-chat-view.types.ts +0 -52
  116. package/src/components/chat/copilot-chat.component.ts +0 -232
  117. package/src/components/copilotkit-tool-render.component.ts +0 -169
  118. package/src/core/__tests__/copilotkit.service.spec.ts +0 -1051
  119. package/src/core/__tests__/copilotkit.service.wildcard.spec.ts +0 -316
  120. package/src/core/chat-configuration/__tests__/chat-configuration.service.spec.ts +0 -287
  121. package/src/core/chat-configuration/chat-configuration.providers.ts +0 -71
  122. package/src/core/chat-configuration/chat-configuration.service.ts +0 -162
  123. package/src/core/chat-configuration/chat-configuration.types.ts +0 -57
  124. package/src/core/copilotkit.providers.ts +0 -59
  125. package/src/core/copilotkit.service.ts +0 -542
  126. package/src/core/copilotkit.types.ts +0 -132
  127. package/src/directives/__tests__/copilotkit-agent-context.directive.spec.ts +0 -384
  128. package/src/directives/__tests__/copilotkit-agent.directive.spec.ts +0 -253
  129. package/src/directives/__tests__/copilotkit-chat-config.directive.spec.ts +0 -385
  130. package/src/directives/__tests__/copilotkit-config.directive.spec.ts +0 -69
  131. package/src/directives/__tests__/copilotkit-frontend-tool-simple.directive.spec.ts +0 -60
  132. package/src/directives/__tests__/copilotkit-frontend-tool.directive.spec.ts +0 -108
  133. package/src/directives/__tests__/copilotkit-human-in-the-loop.directive.spec.ts +0 -452
  134. package/src/directives/copilotkit-agent-context.directive.ts +0 -138
  135. package/src/directives/copilotkit-agent.directive.ts +0 -225
  136. package/src/directives/copilotkit-chat-config.directive.ts +0 -241
  137. package/src/directives/copilotkit-config.directive.ts +0 -81
  138. package/src/directives/copilotkit-frontend-tool.directive.ts +0 -145
  139. package/src/directives/copilotkit-human-in-the-loop.directive.ts +0 -281
  140. package/src/directives/stick-to-bottom.directive.ts +0 -204
  141. package/src/index.ts +0 -105
  142. package/src/lib/directives/tooltip.directive.ts +0 -292
  143. package/src/lib/slots/__tests__/slot.utils.spec.ts +0 -377
  144. package/src/lib/slots/copilot-slot.component.ts +0 -135
  145. package/src/lib/slots/index.ts +0 -3
  146. package/src/lib/slots/slot.types.ts +0 -64
  147. package/src/lib/slots/slot.utils.ts +0 -289
  148. package/src/lib/utils.ts +0 -10
  149. package/src/public-api.ts +0 -1
  150. package/src/services/resize-observer.service.ts +0 -181
  151. package/src/services/scroll-position.service.ts +0 -169
  152. package/src/styles/globals.css +0 -266
  153. package/src/styles/index.css +0 -3
  154. package/src/test-setup.ts +0 -15
  155. package/src/testing/index.ts +0 -3
  156. package/src/testing/testing.utils.ts +0 -248
  157. package/src/types/frontend-tool.ts +0 -44
  158. package/src/types/human-in-the-loop.ts +0 -52
  159. package/src/utils/__tests__/agent.utils.spec.ts +0 -234
  160. package/src/utils/__tests__/chat-config.utils.spec.ts +0 -306
  161. package/src/utils/__tests__/frontend-tool-inject.spec.ts +0 -350
  162. package/src/utils/__tests__/frontend-tool-integration.spec.ts +0 -199
  163. package/src/utils/__tests__/frontend-tool.utils.spec.ts +0 -272
  164. package/src/utils/__tests__/human-in-the-loop.utils.spec.ts +0 -365
  165. package/src/utils/agent-context.utils.ts +0 -133
  166. package/src/utils/agent.utils.ts +0 -239
  167. package/src/utils/chat-config.utils.ts +0 -221
  168. package/src/utils/copilotkit.utils.ts +0 -20
  169. package/src/utils/frontend-tool.utils.ts +0 -266
  170. package/src/utils/human-in-the-loop.utils.ts +0 -359
  171. package/tsconfig.spec.json +0 -12
@@ -0,0 +1,384 @@
1
+ import { Component, Input, Output, EventEmitter, signal, computed, ChangeDetectionStrategy, ViewEncapsulation, inject } from '@angular/core';
2
+ import { CommonModule } from '@angular/common';
3
+ import { LucideAngularModule, Copy, Check, ThumbsUp, ThumbsDown, Volume2, RefreshCw } from 'lucide-angular';
4
+ import { CopilotTooltipDirective } from '../../lib/directives/tooltip.directive';
5
+ import { CopilotChatConfigurationService } from '../../core/chat-configuration/chat-configuration.service';
6
+ import { cn } from '../../lib/utils';
7
+ import * as i0 from "@angular/core";
8
+ import * as i1 from "../../lib/directives/tooltip.directive";
9
+ import * as i2 from "lucide-angular";
10
+ // Base toolbar button component
11
+ export class CopilotChatAssistantMessageToolbarButtonComponent {
12
+ title = '';
13
+ disabled = false;
14
+ set inputClass(value) {
15
+ this.customClass.set(value);
16
+ }
17
+ customClass = signal(undefined);
18
+ computedClass = computed(() => {
19
+ return cn(
20
+ // Flex centering with gap (from React button base styles)
21
+ 'inline-flex items-center justify-center gap-2',
22
+ // Cursor
23
+ 'cursor-pointer',
24
+ // Background and text
25
+ 'p-0 text-[rgb(93,93,93)] hover:bg-[#E8E8E8]',
26
+ // Dark mode
27
+ 'dark:text-[rgb(243,243,243)] dark:hover:bg-[#303030]',
28
+ // Shape and sizing
29
+ 'h-8 w-8 rounded-md',
30
+ // Interactions
31
+ 'transition-colors',
32
+ // Hover states
33
+ 'hover:text-[rgb(93,93,93)]', 'dark:hover:text-[rgb(243,243,243)]',
34
+ // Focus states
35
+ 'focus:outline-none focus:ring-2 focus:ring-offset-2',
36
+ // Disabled state
37
+ 'disabled:opacity-50 disabled:cursor-not-allowed',
38
+ // SVG styling from React Button component
39
+ '[&_svg]:pointer-events-none [&_svg]:shrink-0',
40
+ // Ensure proper sizing
41
+ 'shrink-0', this.customClass());
42
+ });
43
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CopilotChatAssistantMessageToolbarButtonComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
44
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: CopilotChatAssistantMessageToolbarButtonComponent, isStandalone: true, selector: "button[copilotChatAssistantMessageToolbarButton]", inputs: { title: "title", disabled: "disabled", inputClass: "inputClass" }, host: { attributes: { "type": "button" }, properties: { "class": "computedClass()", "attr.disabled": "disabled ? true : null", "attr.aria-label": "title" } }, hostDirectives: [{ directive: i1.CopilotTooltipDirective, inputs: ["copilotTooltip", "title", "tooltipPosition", "tooltipPosition", "tooltipDelay", "tooltipDelay"] }], ngImport: i0, template: `
45
+ <ng-content></ng-content>
46
+ `, isInline: true, dependencies: [{ kind: "ngmodule", type: CommonModule }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None });
47
+ }
48
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CopilotChatAssistantMessageToolbarButtonComponent, decorators: [{
49
+ type: Component,
50
+ args: [{
51
+ selector: 'button[copilotChatAssistantMessageToolbarButton]',
52
+ standalone: true,
53
+ imports: [CommonModule],
54
+ changeDetection: ChangeDetectionStrategy.OnPush,
55
+ encapsulation: ViewEncapsulation.None,
56
+ template: `
57
+ <ng-content></ng-content>
58
+ `,
59
+ host: {
60
+ '[class]': 'computedClass()',
61
+ '[attr.disabled]': 'disabled ? true : null',
62
+ 'type': 'button',
63
+ '[attr.aria-label]': 'title'
64
+ },
65
+ hostDirectives: [
66
+ {
67
+ directive: CopilotTooltipDirective,
68
+ inputs: ['copilotTooltip: title', 'tooltipPosition', 'tooltipDelay']
69
+ }
70
+ ]
71
+ }]
72
+ }], propDecorators: { title: [{
73
+ type: Input
74
+ }], disabled: [{
75
+ type: Input
76
+ }], inputClass: [{
77
+ type: Input
78
+ }] } });
79
+ // Copy button component
80
+ export class CopilotChatAssistantMessageCopyButtonComponent {
81
+ title;
82
+ disabled = false;
83
+ inputClass;
84
+ content;
85
+ clicked = new EventEmitter();
86
+ CopyIcon = Copy;
87
+ CheckIcon = Check;
88
+ copied = signal(false);
89
+ chatConfig = inject(CopilotChatConfigurationService);
90
+ get labels() {
91
+ return this.chatConfig.labels();
92
+ }
93
+ handleCopy(event) {
94
+ event?.stopPropagation();
95
+ if (!this.content)
96
+ return;
97
+ // Set copied immediately for instant feedback
98
+ this.copied.set(true);
99
+ setTimeout(() => this.copied.set(false), 2000);
100
+ // Copy to clipboard (fire and forget)
101
+ navigator.clipboard.writeText(this.content).then(() => this.clicked.emit(), (err) => {
102
+ console.error('Failed to copy message:', err);
103
+ this.copied.set(false);
104
+ });
105
+ }
106
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CopilotChatAssistantMessageCopyButtonComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
107
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: CopilotChatAssistantMessageCopyButtonComponent, isStandalone: true, selector: "copilot-chat-assistant-message-copy-button", inputs: { title: "title", disabled: "disabled", inputClass: "inputClass", content: "content" }, outputs: { clicked: "clicked" }, ngImport: i0, template: `
108
+ <button
109
+ copilotChatAssistantMessageToolbarButton
110
+ [title]="title || labels.assistantMessageToolbarCopyMessageLabel"
111
+ [disabled]="disabled"
112
+ [inputClass]="inputClass"
113
+ (click)="handleCopy($event)">
114
+ @if (copied()) {
115
+ <lucide-angular [img]="CheckIcon" [size]="18"></lucide-angular>
116
+ } @else {
117
+ <lucide-angular [img]="CopyIcon" [size]="18"></lucide-angular>
118
+ }
119
+ </button>
120
+ `, isInline: true, dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "ngmodule", type: LucideAngularModule }, { kind: "component", type: i2.LucideAngularComponent, selector: "lucide-angular, lucide-icon, i-lucide, span-lucide", inputs: ["class", "name", "img", "color", "absoluteStrokeWidth", "size", "strokeWidth"] }, { kind: "component", type: CopilotChatAssistantMessageToolbarButtonComponent, selector: "button[copilotChatAssistantMessageToolbarButton]", inputs: ["title", "disabled", "inputClass"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None });
121
+ }
122
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CopilotChatAssistantMessageCopyButtonComponent, decorators: [{
123
+ type: Component,
124
+ args: [{
125
+ selector: 'copilot-chat-assistant-message-copy-button',
126
+ standalone: true,
127
+ imports: [CommonModule, LucideAngularModule, CopilotChatAssistantMessageToolbarButtonComponent],
128
+ changeDetection: ChangeDetectionStrategy.OnPush,
129
+ encapsulation: ViewEncapsulation.None,
130
+ template: `
131
+ <button
132
+ copilotChatAssistantMessageToolbarButton
133
+ [title]="title || labels.assistantMessageToolbarCopyMessageLabel"
134
+ [disabled]="disabled"
135
+ [inputClass]="inputClass"
136
+ (click)="handleCopy($event)">
137
+ @if (copied()) {
138
+ <lucide-angular [img]="CheckIcon" [size]="18"></lucide-angular>
139
+ } @else {
140
+ <lucide-angular [img]="CopyIcon" [size]="18"></lucide-angular>
141
+ }
142
+ </button>
143
+ `
144
+ }]
145
+ }], propDecorators: { title: [{
146
+ type: Input
147
+ }], disabled: [{
148
+ type: Input
149
+ }], inputClass: [{
150
+ type: Input
151
+ }], content: [{
152
+ type: Input
153
+ }], clicked: [{
154
+ type: Output
155
+ }] } });
156
+ // Thumbs up button component
157
+ export class CopilotChatAssistantMessageThumbsUpButtonComponent {
158
+ title;
159
+ disabled = false;
160
+ inputClass;
161
+ clicked = new EventEmitter();
162
+ ThumbsUpIcon = ThumbsUp;
163
+ chatConfig = inject(CopilotChatConfigurationService);
164
+ get labels() {
165
+ return this.chatConfig.labels();
166
+ }
167
+ handleClick(event) {
168
+ event?.stopPropagation();
169
+ if (!this.disabled) {
170
+ this.clicked.emit();
171
+ }
172
+ }
173
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CopilotChatAssistantMessageThumbsUpButtonComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
174
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: CopilotChatAssistantMessageThumbsUpButtonComponent, isStandalone: true, selector: "copilot-chat-assistant-message-thumbs-up-button", inputs: { title: "title", disabled: "disabled", inputClass: "inputClass" }, outputs: { clicked: "clicked" }, ngImport: i0, template: `
175
+ <button
176
+ copilotChatAssistantMessageToolbarButton
177
+ [title]="title || labels.assistantMessageToolbarThumbsUpLabel"
178
+ [disabled]="disabled"
179
+ [inputClass]="inputClass"
180
+ (click)="handleClick($event)">
181
+ <lucide-angular [img]="ThumbsUpIcon" [size]="18"></lucide-angular>
182
+ </button>
183
+ `, isInline: true, dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "ngmodule", type: LucideAngularModule }, { kind: "component", type: i2.LucideAngularComponent, selector: "lucide-angular, lucide-icon, i-lucide, span-lucide", inputs: ["class", "name", "img", "color", "absoluteStrokeWidth", "size", "strokeWidth"] }, { kind: "component", type: CopilotChatAssistantMessageToolbarButtonComponent, selector: "button[copilotChatAssistantMessageToolbarButton]", inputs: ["title", "disabled", "inputClass"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None });
184
+ }
185
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CopilotChatAssistantMessageThumbsUpButtonComponent, decorators: [{
186
+ type: Component,
187
+ args: [{
188
+ selector: 'copilot-chat-assistant-message-thumbs-up-button',
189
+ standalone: true,
190
+ imports: [CommonModule, LucideAngularModule, CopilotChatAssistantMessageToolbarButtonComponent],
191
+ changeDetection: ChangeDetectionStrategy.OnPush,
192
+ encapsulation: ViewEncapsulation.None,
193
+ template: `
194
+ <button
195
+ copilotChatAssistantMessageToolbarButton
196
+ [title]="title || labels.assistantMessageToolbarThumbsUpLabel"
197
+ [disabled]="disabled"
198
+ [inputClass]="inputClass"
199
+ (click)="handleClick($event)">
200
+ <lucide-angular [img]="ThumbsUpIcon" [size]="18"></lucide-angular>
201
+ </button>
202
+ `
203
+ }]
204
+ }], propDecorators: { title: [{
205
+ type: Input
206
+ }], disabled: [{
207
+ type: Input
208
+ }], inputClass: [{
209
+ type: Input
210
+ }], clicked: [{
211
+ type: Output
212
+ }] } });
213
+ // Thumbs down button component
214
+ export class CopilotChatAssistantMessageThumbsDownButtonComponent {
215
+ title;
216
+ disabled = false;
217
+ inputClass;
218
+ clicked = new EventEmitter();
219
+ ThumbsDownIcon = ThumbsDown;
220
+ chatConfig = inject(CopilotChatConfigurationService);
221
+ get labels() {
222
+ return this.chatConfig.labels();
223
+ }
224
+ handleClick(event) {
225
+ event?.stopPropagation();
226
+ if (!this.disabled) {
227
+ this.clicked.emit();
228
+ }
229
+ }
230
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CopilotChatAssistantMessageThumbsDownButtonComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
231
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: CopilotChatAssistantMessageThumbsDownButtonComponent, isStandalone: true, selector: "copilot-chat-assistant-message-thumbs-down-button", inputs: { title: "title", disabled: "disabled", inputClass: "inputClass" }, outputs: { clicked: "clicked" }, ngImport: i0, template: `
232
+ <button
233
+ copilotChatAssistantMessageToolbarButton
234
+ [title]="title || labels.assistantMessageToolbarThumbsDownLabel"
235
+ [disabled]="disabled"
236
+ [inputClass]="inputClass"
237
+ (click)="handleClick($event)">
238
+ <lucide-angular [img]="ThumbsDownIcon" [size]="18"></lucide-angular>
239
+ </button>
240
+ `, isInline: true, dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "ngmodule", type: LucideAngularModule }, { kind: "component", type: i2.LucideAngularComponent, selector: "lucide-angular, lucide-icon, i-lucide, span-lucide", inputs: ["class", "name", "img", "color", "absoluteStrokeWidth", "size", "strokeWidth"] }, { kind: "component", type: CopilotChatAssistantMessageToolbarButtonComponent, selector: "button[copilotChatAssistantMessageToolbarButton]", inputs: ["title", "disabled", "inputClass"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None });
241
+ }
242
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CopilotChatAssistantMessageThumbsDownButtonComponent, decorators: [{
243
+ type: Component,
244
+ args: [{
245
+ selector: 'copilot-chat-assistant-message-thumbs-down-button',
246
+ standalone: true,
247
+ imports: [CommonModule, LucideAngularModule, CopilotChatAssistantMessageToolbarButtonComponent],
248
+ changeDetection: ChangeDetectionStrategy.OnPush,
249
+ encapsulation: ViewEncapsulation.None,
250
+ template: `
251
+ <button
252
+ copilotChatAssistantMessageToolbarButton
253
+ [title]="title || labels.assistantMessageToolbarThumbsDownLabel"
254
+ [disabled]="disabled"
255
+ [inputClass]="inputClass"
256
+ (click)="handleClick($event)">
257
+ <lucide-angular [img]="ThumbsDownIcon" [size]="18"></lucide-angular>
258
+ </button>
259
+ `
260
+ }]
261
+ }], propDecorators: { title: [{
262
+ type: Input
263
+ }], disabled: [{
264
+ type: Input
265
+ }], inputClass: [{
266
+ type: Input
267
+ }], clicked: [{
268
+ type: Output
269
+ }] } });
270
+ // Read aloud button component
271
+ export class CopilotChatAssistantMessageReadAloudButtonComponent {
272
+ title;
273
+ disabled = false;
274
+ inputClass;
275
+ clicked = new EventEmitter();
276
+ Volume2Icon = Volume2;
277
+ chatConfig = inject(CopilotChatConfigurationService);
278
+ get labels() {
279
+ return this.chatConfig.labels();
280
+ }
281
+ handleClick(event) {
282
+ event?.stopPropagation();
283
+ if (!this.disabled) {
284
+ this.clicked.emit();
285
+ }
286
+ }
287
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CopilotChatAssistantMessageReadAloudButtonComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
288
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: CopilotChatAssistantMessageReadAloudButtonComponent, isStandalone: true, selector: "copilot-chat-assistant-message-read-aloud-button", inputs: { title: "title", disabled: "disabled", inputClass: "inputClass" }, outputs: { clicked: "clicked" }, ngImport: i0, template: `
289
+ <button
290
+ copilotChatAssistantMessageToolbarButton
291
+ [title]="title || labels.assistantMessageToolbarReadAloudLabel"
292
+ [disabled]="disabled"
293
+ [inputClass]="inputClass"
294
+ (click)="handleClick($event)">
295
+ <lucide-angular [img]="Volume2Icon" [size]="20"></lucide-angular>
296
+ </button>
297
+ `, isInline: true, dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "ngmodule", type: LucideAngularModule }, { kind: "component", type: i2.LucideAngularComponent, selector: "lucide-angular, lucide-icon, i-lucide, span-lucide", inputs: ["class", "name", "img", "color", "absoluteStrokeWidth", "size", "strokeWidth"] }, { kind: "component", type: CopilotChatAssistantMessageToolbarButtonComponent, selector: "button[copilotChatAssistantMessageToolbarButton]", inputs: ["title", "disabled", "inputClass"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None });
298
+ }
299
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CopilotChatAssistantMessageReadAloudButtonComponent, decorators: [{
300
+ type: Component,
301
+ args: [{
302
+ selector: 'copilot-chat-assistant-message-read-aloud-button',
303
+ standalone: true,
304
+ imports: [CommonModule, LucideAngularModule, CopilotChatAssistantMessageToolbarButtonComponent],
305
+ changeDetection: ChangeDetectionStrategy.OnPush,
306
+ encapsulation: ViewEncapsulation.None,
307
+ template: `
308
+ <button
309
+ copilotChatAssistantMessageToolbarButton
310
+ [title]="title || labels.assistantMessageToolbarReadAloudLabel"
311
+ [disabled]="disabled"
312
+ [inputClass]="inputClass"
313
+ (click)="handleClick($event)">
314
+ <lucide-angular [img]="Volume2Icon" [size]="20"></lucide-angular>
315
+ </button>
316
+ `
317
+ }]
318
+ }], propDecorators: { title: [{
319
+ type: Input
320
+ }], disabled: [{
321
+ type: Input
322
+ }], inputClass: [{
323
+ type: Input
324
+ }], clicked: [{
325
+ type: Output
326
+ }] } });
327
+ // Regenerate button component
328
+ export class CopilotChatAssistantMessageRegenerateButtonComponent {
329
+ title;
330
+ disabled = false;
331
+ inputClass;
332
+ clicked = new EventEmitter();
333
+ RefreshCwIcon = RefreshCw;
334
+ chatConfig = inject(CopilotChatConfigurationService);
335
+ get labels() {
336
+ return this.chatConfig.labels();
337
+ }
338
+ handleClick(event) {
339
+ event?.stopPropagation();
340
+ if (!this.disabled) {
341
+ this.clicked.emit();
342
+ }
343
+ }
344
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CopilotChatAssistantMessageRegenerateButtonComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
345
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: CopilotChatAssistantMessageRegenerateButtonComponent, isStandalone: true, selector: "copilot-chat-assistant-message-regenerate-button", inputs: { title: "title", disabled: "disabled", inputClass: "inputClass" }, outputs: { clicked: "clicked" }, ngImport: i0, template: `
346
+ <button
347
+ copilotChatAssistantMessageToolbarButton
348
+ [title]="title || labels.assistantMessageToolbarRegenerateLabel"
349
+ [disabled]="disabled"
350
+ [inputClass]="inputClass"
351
+ (click)="handleClick($event)">
352
+ <lucide-angular [img]="RefreshCwIcon" [size]="18"></lucide-angular>
353
+ </button>
354
+ `, isInline: true, dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "ngmodule", type: LucideAngularModule }, { kind: "component", type: i2.LucideAngularComponent, selector: "lucide-angular, lucide-icon, i-lucide, span-lucide", inputs: ["class", "name", "img", "color", "absoluteStrokeWidth", "size", "strokeWidth"] }, { kind: "component", type: CopilotChatAssistantMessageToolbarButtonComponent, selector: "button[copilotChatAssistantMessageToolbarButton]", inputs: ["title", "disabled", "inputClass"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None });
355
+ }
356
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CopilotChatAssistantMessageRegenerateButtonComponent, decorators: [{
357
+ type: Component,
358
+ args: [{
359
+ selector: 'copilot-chat-assistant-message-regenerate-button',
360
+ standalone: true,
361
+ imports: [CommonModule, LucideAngularModule, CopilotChatAssistantMessageToolbarButtonComponent],
362
+ changeDetection: ChangeDetectionStrategy.OnPush,
363
+ encapsulation: ViewEncapsulation.None,
364
+ template: `
365
+ <button
366
+ copilotChatAssistantMessageToolbarButton
367
+ [title]="title || labels.assistantMessageToolbarRegenerateLabel"
368
+ [disabled]="disabled"
369
+ [inputClass]="inputClass"
370
+ (click)="handleClick($event)">
371
+ <lucide-angular [img]="RefreshCwIcon" [size]="18"></lucide-angular>
372
+ </button>
373
+ `
374
+ }]
375
+ }], propDecorators: { title: [{
376
+ type: Input
377
+ }], disabled: [{
378
+ type: Input
379
+ }], inputClass: [{
380
+ type: Input
381
+ }], clicked: [{
382
+ type: Output
383
+ }] } });
384
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29waWxvdC1jaGF0LWFzc2lzdGFudC1tZXNzYWdlLWJ1dHRvbnMuY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vc3JjL2NvbXBvbmVudHMvY2hhdC9jb3BpbG90LWNoYXQtYXNzaXN0YW50LW1lc3NhZ2UtYnV0dG9ucy5jb21wb25lbnQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUNMLFNBQVMsRUFDVCxLQUFLLEVBQ0wsTUFBTSxFQUNOLFlBQVksRUFDWixNQUFNLEVBQ04sUUFBUSxFQUNSLHVCQUF1QixFQUN2QixpQkFBaUIsRUFDakIsTUFBTSxFQUNQLE1BQU0sZUFBZSxDQUFDO0FBQ3ZCLE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQUMvQyxPQUFPLEVBQUUsbUJBQW1CLEVBQUUsSUFBSSxFQUFFLEtBQUssRUFBRSxRQUFRLEVBQUUsVUFBVSxFQUFFLE9BQU8sRUFBRSxTQUFTLEVBQUUsTUFBTSxnQkFBZ0IsQ0FBQztBQUM1RyxPQUFPLEVBQUUsdUJBQXVCLEVBQUUsTUFBTSx3Q0FBd0MsQ0FBQztBQUNqRixPQUFPLEVBQUUsK0JBQStCLEVBQUUsTUFBTSwwREFBMEQsQ0FBQztBQUMzRyxPQUFPLEVBQUUsRUFBRSxFQUFFLE1BQU0saUJBQWlCLENBQUM7Ozs7QUFFckMsZ0NBQWdDO0FBdUJoQyxNQUFNLE9BQU8saURBQWlEO0lBQ25ELEtBQUssR0FBRyxFQUFFLENBQUM7SUFDWCxRQUFRLEdBQUcsS0FBSyxDQUFDO0lBQzFCLElBQWEsVUFBVSxDQUFDLEtBQXlCO1FBQy9DLElBQUksQ0FBQyxXQUFXLENBQUMsR0FBRyxDQUFDLEtBQUssQ0FBQyxDQUFDO0lBQzlCLENBQUM7SUFFTyxXQUFXLEdBQUcsTUFBTSxDQUFxQixTQUFTLENBQUMsQ0FBQztJQUU1RCxhQUFhLEdBQUcsUUFBUSxDQUFDLEdBQUcsRUFBRTtRQUM1QixPQUFPLEVBQUU7UUFDUCwwREFB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QUFpRCxDQUFDO29CQUMvRixlQUFlLEVBQUUsdUJBQXVCLENBQUMsTUFBTTtvQkFDL0MsYUFBYSxFQUFFLGlCQUFpQixDQUFDLElBQUk7b0JBQ3JDLFFBQVEsRUFBRTs7Ozs7Ozs7O0dBU1Q7aUJBQ0Y7OEJBRVUsS0FBSztzQkFBYixLQUFLO2dCQUNHLFFBQVE7c0JBQWhCLEtBQUs7Z0JBQ0csVUFBVTtzQkFBbEIsS0FBSztnQkFDSSxPQUFPO3NCQUFoQixNQUFNIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHtcbiAgQ29tcG9uZW50LFxuICBJbnB1dCxcbiAgT3V0cHV0LFxuICBFdmVudEVtaXR0ZXIsXG4gIHNpZ25hbCxcbiAgY29tcHV0ZWQsXG4gIENoYW5nZURldGVjdGlvblN0cmF0ZWd5LFxuICBWaWV3RW5jYXBzdWxhdGlvbixcbiAgaW5qZWN0XG59IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgQ29tbW9uTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvY29tbW9uJztcbmltcG9ydCB7IEx1Y2lkZUFuZ3VsYXJNb2R1bGUsIENvcHksIENoZWNrLCBUaHVtYnNVcCwgVGh1bWJzRG93biwgVm9sdW1lMiwgUmVmcmVzaEN3IH0gZnJvbSAnbHVjaWRlLWFuZ3VsYXInO1xuaW1wb3J0IHsgQ29waWxvdFRvb2x0aXBEaXJlY3RpdmUgfSBmcm9tICcuLi8uLi9saWIvZGlyZWN0aXZlcy90b29sdGlwLmRpcmVjdGl2ZSc7XG5pbXBvcnQgeyBDb3BpbG90Q2hhdENvbmZpZ3VyYXRpb25TZXJ2aWNlIH0gZnJvbSAnLi4vLi4vY29yZS9jaGF0LWNvbmZpZ3VyYXRpb24vY2hhdC1jb25maWd1cmF0aW9uLnNlcnZpY2UnO1xuaW1wb3J0IHsgY24gfSBmcm9tICcuLi8uLi9saWIvdXRpbHMnO1xuXG4vLyBCYXNlIHRvb2xiYXIgYnV0dG9uIGNvbXBvbmVudFxuQENvbXBvbmVudCh7XG4gIHNlbGVjdG9yOiAnYnV0dG9uW2NvcGlsb3RDaGF0QXNzaXN0YW50TWVzc2FnZVRvb2xiYXJCdXR0b25dJyxcbiAgc3RhbmRhbG9uZTogdHJ1ZSxcbiAgaW1wb3J0czogW0NvbW1vbk1vZHVsZV0sXG4gIGNoYW5nZURldGVjdGlvbjogQ2hhbmdlRGV0ZWN0aW9uU3RyYXRlZ3kuT25QdXNoLFxuICBlbmNhcHN1bGF0aW9uOiBWaWV3RW5jYXBzdWxhdGlvbi5Ob25lLFxuICB0ZW1wbGF0ZTogYFxuICAgIDxuZy1jb250ZW50PjwvbmctY29udGVudD5cbiAgYCxcbiAgaG9zdDoge1xuICAgICdbY2xhc3NdJzogJ2NvbXB1dGVkQ2xhc3MoKScsXG4gICAgJ1thdHRyLmRpc2FibGVkXSc6ICdkaXNhYmxlZCA/IHRydWUgOiBudWxsJyxcbiAgICAndHlwZSc6ICdidXR0b24nLFxuICAgICdbYXR0ci5hcmlhLWxhYmVsXSc6ICd0aXRsZSdcbiAgfSxcbiAgaG9zdERpcmVjdGl2ZXM6IFtcbiAgICB7XG4gICAgICBkaXJlY3RpdmU6IENvcGlsb3RUb29sdGlwRGlyZWN0aXZlLFxuICAgICAgaW5wdXRzOiBbJ2NvcGlsb3RUb29sdGlwOiB0aXRsZScsICd0b29sdGlwUG9zaXRpb24nLCAndG9vbHRpcERlbGF5J11cbiAgICB9XG4gIF1cbn0pXG5leHBvcnQgY2xhc3MgQ29waWxvdENoYXRBc3Npc3RhbnRNZXNzYWdlVG9vbGJhckJ1dHRvbkNvbXBvbmVudCB7XG4gIEBJbnB1dCgpIHRpdGxlID0gJyc7XG4gIEBJbnB1dCgpIGRpc2FibGVkID0gZmFsc2U7XG4gIEBJbnB1dCgpIHNldCBpbnB1dENsYXNzKHZhbHVlOiBzdHJpbmcgfCB1bmRlZmluZWQpIHtcbiAgICB0aGlzLmN1c3RvbUNsYXNzLnNldCh2YWx1ZSk7XG4gIH1cbiAgXG4gIHByaXZhdGUgY3VzdG9tQ2xhc3MgPSBzaWduYWw8c3RyaW5nIHwgdW5kZWZpbmVkPih1bmRlZmluZWQpO1xuICBcbiAgY29tcHV0ZWRDbGFzcyA9IGNvbXB1dGVkKCgpID0+IHtcbiAgICByZXR1cm4gY24oXG4gICAgICAvLyBGbGV4IGNlbnRlcmluZyB3aXRoIGdhcCAoZnJvbSBSZWFjdCBidXR0b24gYmFzZSBzdHlsZXMpXG4gICAgICAnaW5saW5lLWZsZXggaXRlbXMtY2VudGVyIGp1c3RpZnktY2VudGVyIGdhcC0yJyxcbiAgICAgIC8vIEN1cnNvclxuICAgICAgJ2N1cnNvci1wb2ludGVyJyxcbiAgICAgIC8vIEJhY2tncm91bmQgYW5kIHRleHRcbiAgICAgICdwLTAgdGV4dC1bcmdiKDkzLDkzLDkzKV0gaG92ZXI6YmctWyNFOEU4RThdJyxcbiAgICAgIC8vIERhcmsgbW9kZVxuICAgICAgJ2Rhcms6dGV4dC1bcmdiKDI0MywyNDMsMjQzKV0gZGFyazpob3ZlcjpiZy1bIzMwMzAzMF0nLFxuICAgICAgLy8gU2hhcGUgYW5kIHNpemluZ1xuICAgICAgJ2gtOCB3LTggcm91bmRlZC1tZCcsXG4gICAgICAvLyBJbnRlcmFjdGlvbnNcbiAgICAgICd0cmFuc2l0aW9uLWNvbG9ycycsXG4gICAgICAvLyBIb3ZlciBzdGF0ZXNcbiAgICAgICdob3Zlcjp0ZXh0LVtyZ2IoOTMsOTMsOTMpXScsXG4gICAgICAnZGFyazpob3Zlcjp0ZXh0LVtyZ2IoMjQzLDI0MywyNDMpXScsXG4gICAgICAvLyBGb2N1cyBzdGF0ZXNcbiAgICAgICdmb2N1czpvdXRsaW5lLW5vbmUgZm9jdXM6cmluZy0yIGZvY3VzOnJpbmctb2Zmc2V0LTInLFxuICAgICAgLy8gRGlzYWJsZWQgc3RhdGVcbiAgICAgICdkaXNhYmxlZDpvcGFjaXR5LTUwIGRpc2FibGVkOmN1cnNvci1ub3QtYWxsb3dlZCcsXG4gICAgICAvLyBTVkcgc3R5bGluZyBmcm9tIFJlYWN0IEJ1dHRvbiBjb21wb25lbnRcbiAgICAgICdbJl9zdmddOnBvaW50ZXItZXZlbnRzLW5vbmUgWyZfc3ZnXTpzaHJpbmstMCcsXG4gICAgICAvLyBFbnN1cmUgcHJvcGVyIHNpemluZ1xuICAgICAgJ3Nocmluay0wJyxcbiAgICAgIHRoaXMuY3VzdG9tQ2xhc3MoKVxuICAgICk7XG4gIH0pO1xufVxuXG4vLyBDb3B5IGJ1dHRvbiBjb21wb25lbnRcbkBDb21wb25lbnQoe1xuICBzZWxlY3RvcjogJ2NvcGlsb3QtY2hhdC1hc3Npc3RhbnQtbWVzc2FnZS1jb3B5LWJ1dHRvbicsXG4gIHN0YW5kYWxvbmU6IHRydWUsXG4gIGltcG9ydHM6IFtDb21tb25Nb2R1bGUsIEx1Y2lkZUFuZ3VsYXJNb2R1bGUsIENvcGlsb3RDaGF0QXNzaXN0YW50TWVzc2FnZVRvb2xiYXJCdXR0b25Db21wb25lbnRdLFxuICBjaGFuZ2VEZXRlY3Rpb246IENoYW5nZURldGVjdGlvblN0cmF0ZWd5Lk9uUHVzaCxcbiAgZW5jYXBzdWxhdGlvbjogVmlld0VuY2Fwc3VsYXRpb24uTm9uZSxcbiAgdGVtcGxhdGU6IGBcbiAgICA8YnV0dG9uIFxuICAgICAgY29waWxvdENoYXRBc3Npc3RhbnRNZXNzYWdlVG9vbGJhckJ1dHRvblxuICAgICAgW3RpdGxlXT1cInRpdGxlIHx8IGxhYmVscy5hc3Npc3RhbnRNZXNzYWdlVG9vbGJhckNvcHlNZXNzYWdlTGFiZWxcIlxuICAgICAgW2Rpc2FibGVkXT1cImRpc2FibGVkXCJcbiAgICAgIFtpbnB1dENsYXNzXT1cImlucHV0Q2xhc3NcIlxuICAgICAgKGNsaWNrKT1cImhhbmRsZUNvcHkoJGV2ZW50KVwiPlxuICAgICAgQGlmIChjb3BpZWQoKSkge1xuICAgICAgICA8bHVjaWRlLWFuZ3VsYXIgW2ltZ109XCJDaGVja0ljb25cIiBbc2l6ZV09XCIxOFwiPjwvbHVjaWRlLWFuZ3VsYXI+XG4gICAgICB9IEBlbHNlIHtcbiAgICAgICAgPGx1Y2lkZS1hbmd1bGFyIFtpbWddPVwiQ29weUljb25cIiBbc2l6ZV09XCIxOFwiPjwvbHVjaWRlLWFuZ3VsYXI+XG4gICAgICB9XG4gICAgPC9idXR0b24+XG4gIGBcbn0pXG5leHBvcnQgY2xhc3MgQ29waWxvdENoYXRBc3Npc3RhbnRNZXNzYWdlQ29weUJ1dHRvbkNvbXBvbmVudCB7XG4gIEBJbnB1dCgpIHRpdGxlPzogc3RyaW5nO1xuICBASW5wdXQoKSBkaXNhYmxlZCA9IGZhbHNlO1xuICBASW5wdXQoKSBpbnB1dENsYXNzPzogc3RyaW5nO1xuICBASW5wdXQoKSBjb250ZW50Pzogc3RyaW5nO1xuICBAT3V0cHV0KCkgY2xpY2tlZCA9IG5ldyBFdmVudEVtaXR0ZXI8dm9pZD4oKTtcbiAgXG4gIHJlYWRvbmx5IENvcHlJY29uID0gQ29weTtcbiAgcmVhZG9ubHkgQ2hlY2tJY29uID0gQ2hlY2s7XG4gIFxuICBjb3BpZWQgPSBzaWduYWwoZmFsc2UpO1xuICBwcml2YXRlIGNoYXRDb25maWcgPSBpbmplY3QoQ29waWxvdENoYXRDb25maWd1cmF0aW9uU2VydmljZSk7XG4gIFxuICBnZXQgbGFiZWxzKCkge1xuICAgIHJldHVybiB0aGlzLmNoYXRDb25maWcubGFiZWxzKCk7XG4gIH1cbiAgXG4gIGhhbmRsZUNvcHkoZXZlbnQ/OiBFdmVudCk6IHZvaWQge1xuICAgIGV2ZW50Py5zdG9wUHJvcGFnYXRpb24oKTtcbiAgICBpZiAoIXRoaXMuY29udGVudCkgcmV0dXJuO1xuICAgIFxuICAgIC8vIFNldCBjb3BpZWQgaW1tZWRpYXRlbHkgZm9yIGluc3RhbnQgZmVlZGJhY2tcbiAgICB0aGlzLmNvcGllZC5zZXQodHJ1ZSk7XG4gICAgc2V0VGltZW91dCgoKSA9PiB0aGlzLmNvcGllZC5zZXQoZmFsc2UpLCAyMDAwKTtcbiAgICBcbiAgICAvLyBDb3B5IHRvIGNsaXBib2FyZCAoZmlyZSBhbmQgZm9yZ2V0KVxuICAgIG5hdmlnYXRvci5jbGlwYm9hcmQud3JpdGVUZXh0KHRoaXMuY29udGVudCkudGhlbihcbiAgICAgICgpID0+IHRoaXMuY2xpY2tlZC5lbWl0KCksXG4gICAgICAoZXJyKSA9PiB7XG4gICAgICAgIGNvbnNvbGUuZXJyb3IoJ0ZhaWxlZCB0byBjb3B5IG1lc3NhZ2U6JywgZXJyKTtcbiAgICAgICAgdGhpcy5jb3BpZWQuc2V0KGZhbHNlKTtcbiAgICAgIH1cbiAgICApO1xuICB9XG59XG5cbi8vIFRodW1icyB1cCBidXR0b24gY29tcG9uZW50XG5AQ29tcG9uZW50KHtcbiAgc2VsZWN0b3I6ICdjb3BpbG90LWNoYXQtYXNzaXN0YW50LW1lc3NhZ2UtdGh1bWJzLXVwLWJ1dHRvbicsXG4gIHN0YW5kYWxvbmU6IHRydWUsXG4gIGltcG9ydHM6IFtDb21tb25Nb2R1bGUsIEx1Y2lkZUFuZ3VsYXJNb2R1bGUsIENvcGlsb3RDaGF0QXNzaXN0YW50TWVzc2FnZVRvb2xiYXJCdXR0b25Db21wb25lbnRdLFxuICBjaGFuZ2VEZXRlY3Rpb246IENoYW5nZURldGVjdGlvblN0cmF0ZWd5Lk9uUHVzaCxcbiAgZW5jYXBzdWxhdGlvbjogVmlld0VuY2Fwc3VsYXRpb24uTm9uZSxcbiAgdGVtcGxhdGU6IGBcbiAgICA8YnV0dG9uIFxuICAgICAgY29waWxvdENoYXRBc3Npc3RhbnRNZXNzYWdlVG9vbGJhckJ1dHRvblxuICAgICAgW3RpdGxlXT1cInRpdGxlIHx8IGxhYmVscy5hc3Npc3RhbnRNZXNzYWdlVG9vbGJhclRodW1ic1VwTGFiZWxcIlxuICAgICAgW2Rpc2FibGVkXT1cImRpc2FibGVkXCJcbiAgICAgIFtpbnB1dENsYXNzXT1cImlucHV0Q2xhc3NcIlxuICAgICAgKGNsaWNrKT1cImhhbmRsZUNsaWNrKCRldmVudClcIj5cbiAgICAgIDxsdWNpZGUtYW5ndWxhciBbaW1nXT1cIlRodW1ic1VwSWNvblwiIFtzaXplXT1cIjE4XCI+PC9sdWNpZGUtYW5ndWxhcj5cbiAgICA8L2J1dHRvbj5cbiAgYFxufSlcbmV4cG9ydCBjbGFzcyBDb3BpbG90Q2hhdEFzc2lzdGFudE1lc3NhZ2VUaHVtYnNVcEJ1dHRvbkNvbXBvbmVudCB7XG4gIEBJbnB1dCgpIHRpdGxlPzogc3RyaW5nO1xuICBASW5wdXQoKSBkaXNhYmxlZCA9IGZhbHNlO1xuICBASW5wdXQoKSBpbnB1dENsYXNzPzogc3RyaW5nO1xuICBAT3V0cHV0KCkgY2xpY2tlZCA9IG5ldyBFdmVudEVtaXR0ZXI8dm9pZD4oKTtcbiAgXG4gIHJlYWRvbmx5IFRodW1ic1VwSWNvbiA9IFRodW1ic1VwO1xuICBwcml2YXRlIGNoYXRDb25maWcgPSBpbmplY3QoQ29waWxvdENoYXRDb25maWd1cmF0aW9uU2VydmljZSk7XG4gIFxuICBnZXQgbGFiZWxzKCkge1xuICAgIHJldHVybiB0aGlzLmNoYXRDb25maWcubGFiZWxzKCk7XG4gIH1cbiAgXG4gIGhhbmRsZUNsaWNrKGV2ZW50PzogRXZlbnQpOiB2b2lkIHtcbiAgICBldmVudD8uc3RvcFByb3BhZ2F0aW9uKCk7XG4gICAgaWYgKCF0aGlzLmRpc2FibGVkKSB7XG4gICAgICB0aGlzLmNsaWNrZWQuZW1pdCgpO1xuICAgIH1cbiAgfVxufVxuXG4vLyBUaHVtYnMgZG93biBidXR0b24gY29tcG9uZW50XG5AQ29tcG9uZW50KHtcbiAgc2VsZWN0b3I6ICdjb3BpbG90LWNoYXQtYXNzaXN0YW50LW1lc3NhZ2UtdGh1bWJzLWRvd24tYnV0dG9uJyxcbiAgc3RhbmRhbG9uZTogdHJ1ZSxcbiAgaW1wb3J0czogW0NvbW1vbk1vZHVsZSwgTHVjaWRlQW5ndWxhck1vZHVsZSwgQ29waWxvdENoYXRBc3Npc3RhbnRNZXNzYWdlVG9vbGJhckJ1dHRvbkNvbXBvbmVudF0sXG4gIGNoYW5nZURldGVjdGlvbjogQ2hhbmdlRGV0ZWN0aW9uU3RyYXRlZ3kuT25QdXNoLFxuICBlbmNhcHN1bGF0aW9uOiBWaWV3RW5jYXBzdWxhdGlvbi5Ob25lLFxuICB0ZW1wbGF0ZTogYFxuICAgIDxidXR0b24gXG4gICAgICBjb3BpbG90Q2hhdEFzc2lzdGFudE1lc3NhZ2VUb29sYmFyQnV0dG9uXG4gICAgICBbdGl0bGVdPVwidGl0bGUgfHwgbGFiZWxzLmFzc2lzdGFudE1lc3NhZ2VUb29sYmFyVGh1bWJzRG93bkxhYmVsXCJcbiAgICAgIFtkaXNhYmxlZF09XCJkaXNhYmxlZFwiXG4gICAgICBbaW5wdXRDbGFzc109XCJpbnB1dENsYXNzXCJcbiAgICAgIChjbGljayk9XCJoYW5kbGVDbGljaygkZXZlbnQpXCI+XG4gICAgICA8bHVjaWRlLWFuZ3VsYXIgW2ltZ109XCJUaHVtYnNEb3duSWNvblwiIFtzaXplXT1cIjE4XCI+PC9sdWNpZGUtYW5ndWxhcj5cbiAgICA8L2J1dHRvbj5cbiAgYFxufSlcbmV4cG9ydCBjbGFzcyBDb3BpbG90Q2hhdEFzc2lzdGFudE1lc3NhZ2VUaHVtYnNEb3duQnV0dG9uQ29tcG9uZW50IHtcbiAgQElucHV0KCkgdGl0bGU/OiBzdHJpbmc7XG4gIEBJbnB1dCgpIGRpc2FibGVkID0gZmFsc2U7XG4gIEBJbnB1dCgpIGlucHV0Q2xhc3M/OiBzdHJpbmc7XG4gIEBPdXRwdXQoKSBjbGlja2VkID0gbmV3IEV2ZW50RW1pdHRlcjx2b2lkPigpO1xuICBcbiAgcmVhZG9ubHkgVGh1bWJzRG93bkljb24gPSBUaHVtYnNEb3duO1xuICBwcml2YXRlIGNoYXRDb25maWcgPSBpbmplY3QoQ29waWxvdENoYXRDb25maWd1cmF0aW9uU2VydmljZSk7XG4gIFxuICBnZXQgbGFiZWxzKCkge1xuICAgIHJldHVybiB0aGlzLmNoYXRDb25maWcubGFiZWxzKCk7XG4gIH1cbiAgXG4gIGhhbmRsZUNsaWNrKGV2ZW50PzogRXZlbnQpOiB2b2lkIHtcbiAgICBldmVudD8uc3RvcFByb3BhZ2F0aW9uKCk7XG4gICAgaWYgKCF0aGlzLmRpc2FibGVkKSB7XG4gICAgICB0aGlzLmNsaWNrZWQuZW1pdCgpO1xuICAgIH1cbiAgfVxufVxuXG4vLyBSZWFkIGFsb3VkIGJ1dHRvbiBjb21wb25lbnRcbkBDb21wb25lbnQoe1xuICBzZWxlY3RvcjogJ2NvcGlsb3QtY2hhdC1hc3Npc3RhbnQtbWVzc2FnZS1yZWFkLWFsb3VkLWJ1dHRvbicsXG4gIHN0YW5kYWxvbmU6IHRydWUsXG4gIGltcG9ydHM6IFtDb21tb25Nb2R1bGUsIEx1Y2lkZUFuZ3VsYXJNb2R1bGUsIENvcGlsb3RDaGF0QXNzaXN0YW50TWVzc2FnZVRvb2xiYXJCdXR0b25Db21wb25lbnRdLFxuICBjaGFuZ2VEZXRlY3Rpb246IENoYW5nZURldGVjdGlvblN0cmF0ZWd5Lk9uUHVzaCxcbiAgZW5jYXBzdWxhdGlvbjogVmlld0VuY2Fwc3VsYXRpb24uTm9uZSxcbiAgdGVtcGxhdGU6IGBcbiAgICA8YnV0dG9uIFxuICAgICAgY29waWxvdENoYXRBc3Npc3RhbnRNZXNzYWdlVG9vbGJhckJ1dHRvblxuICAgICAgW3RpdGxlXT1cInRpdGxlIHx8IGxhYmVscy5hc3Npc3RhbnRNZXNzYWdlVG9vbGJhclJlYWRBbG91ZExhYmVsXCJcbiAgICAgIFtkaXNhYmxlZF09XCJkaXNhYmxlZFwiXG4gICAgICBbaW5wdXRDbGFzc109XCJpbnB1dENsYXNzXCJcbiAgICAgIChjbGljayk9XCJoYW5kbGVDbGljaygkZXZlbnQpXCI+XG4gICAgICA8bHVjaWRlLWFuZ3VsYXIgW2ltZ109XCJWb2x1bWUySWNvblwiIFtzaXplXT1cIjIwXCI+PC9sdWNpZGUtYW5ndWxhcj5cbiAgICA8L2J1dHRvbj5cbiAgYFxufSlcbmV4cG9ydCBjbGFzcyBDb3BpbG90Q2hhdEFzc2lzdGFudE1lc3NhZ2VSZWFkQWxvdWRCdXR0b25Db21wb25lbnQge1xuICBASW5wdXQoKSB0aXRsZT86IHN0cmluZztcbiAgQElucHV0KCkgZGlzYWJsZWQgPSBmYWxzZTtcbiAgQElucHV0KCkgaW5wdXRDbGFzcz86IHN0cmluZztcbiAgQE91dHB1dCgpIGNsaWNrZWQgPSBuZXcgRXZlbnRFbWl0dGVyPHZvaWQ+KCk7XG4gIFxuICByZWFkb25seSBWb2x1bWUySWNvbiA9IFZvbHVtZTI7XG4gIHByaXZhdGUgY2hhdENvbmZpZyA9IGluamVjdChDb3BpbG90Q2hhdENvbmZpZ3VyYXRpb25TZXJ2aWNlKTtcbiAgXG4gIGdldCBsYWJlbHMoKSB7XG4gICAgcmV0dXJuIHRoaXMuY2hhdENvbmZpZy5sYWJlbHMoKTtcbiAgfVxuICBcbiAgaGFuZGxlQ2xpY2soZXZlbnQ/OiBFdmVudCk6IHZvaWQge1xuICAgIGV2ZW50Py5zdG9wUHJvcGFnYXRpb24oKTtcbiAgICBpZiAoIXRoaXMuZGlzYWJsZWQpIHtcbiAgICAgIHRoaXMuY2xpY2tlZC5lbWl0KCk7XG4gICAgfVxuICB9XG59XG5cbi8vIFJlZ2VuZXJhdGUgYnV0dG9uIGNvbXBvbmVudFxuQENvbXBvbmVudCh7XG4gIHNlbGVjdG9yOiAnY29waWxvdC1jaGF0LWFzc2lzdGFudC1tZXNzYWdlLXJlZ2VuZXJhdGUtYnV0dG9uJyxcbiAgc3RhbmRhbG9uZTogdHJ1ZSxcbiAgaW1wb3J0czogW0NvbW1vbk1vZHVsZSwgTHVjaWRlQW5ndWxhck1vZHVsZSwgQ29waWxvdENoYXRBc3Npc3RhbnRNZXNzYWdlVG9vbGJhckJ1dHRvbkNvbXBvbmVudF0sXG4gIGNoYW5nZURldGVjdGlvbjogQ2hhbmdlRGV0ZWN0aW9uU3RyYXRlZ3kuT25QdXNoLFxuICBlbmNhcHN1bGF0aW9uOiBWaWV3RW5jYXBzdWxhdGlvbi5Ob25lLFxuICB0ZW1wbGF0ZTogYFxuICAgIDxidXR0b24gXG4gICAgICBjb3BpbG90Q2hhdEFzc2lzdGFudE1lc3NhZ2VUb29sYmFyQnV0dG9uXG4gICAgICBbdGl0bGVdPVwidGl0bGUgfHwgbGFiZWxzLmFzc2lzdGFudE1lc3NhZ2VUb29sYmFyUmVnZW5lcmF0ZUxhYmVsXCJcbiAgICAgIFtkaXNhYmxlZF09XCJkaXNhYmxlZFwiXG4gICAgICBbaW5wdXRDbGFzc109XCJpbnB1dENsYXNzXCJcbiAgICAgIChjbGljayk9XCJoYW5kbGVDbGljaygkZXZlbnQpXCI+XG4gICAgICA8bHVjaWRlLWFuZ3VsYXIgW2ltZ109XCJSZWZyZXNoQ3dJY29uXCIgW3NpemVdPVwiMThcIj48L2x1Y2lkZS1hbmd1bGFyPlxuICAgIDwvYnV0dG9uPlxuICBgXG59KVxuZXhwb3J0IGNsYXNzIENvcGlsb3RDaGF0QXNzaXN0YW50TWVzc2FnZVJlZ2VuZXJhdGVCdXR0b25Db21wb25lbnQge1xuICBASW5wdXQoKSB0aXRsZT86IHN0cmluZztcbiAgQElucHV0KCkgZGlzYWJsZWQgPSBmYWxzZTtcbiAgQElucHV0KCkgaW5wdXRDbGFzcz86IHN0cmluZztcbiAgQE91dHB1dCgpIGNsaWNrZWQgPSBuZXcgRXZlbnRFbWl0dGVyPHZvaWQ+KCk7XG4gIFxuICByZWFkb25seSBSZWZyZXNoQ3dJY29uID0gUmVmcmVzaEN3O1xuICBwcml2YXRlIGNoYXRDb25maWcgPSBpbmplY3QoQ29waWxvdENoYXRDb25maWd1cmF0aW9uU2VydmljZSk7XG4gIFxuICBnZXQgbGFiZWxzKCkge1xuICAgIHJldHVybiB0aGlzLmNoYXRDb25maWcubGFiZWxzKCk7XG4gIH1cbiAgXG4gIGhhbmRsZUNsaWNrKGV2ZW50PzogRXZlbnQpOiB2b2lkIHtcbiAgICBldmVudD8uc3RvcFByb3BhZ2F0aW9uKCk7XG4gICAgaWYgKCF0aGlzLmRpc2FibGVkKSB7XG4gICAgICB0aGlzLmNsaWNrZWQuZW1pdCgpO1xuICAgIH1cbiAgfVxufVxuIl19