cloud-ide-element 0.0.1 → 1.0.0

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 (50) hide show
  1. package/README.md +271 -24
  2. package/esm2022/lib/components/confirmation-modal/confirmation-modal.component.mjs +182 -0
  3. package/esm2022/lib/components/data-grid/data-grid.component.mjs +1363 -0
  4. package/esm2022/lib/components/data-grid/data-grid.types.mjs +37 -0
  5. package/esm2022/lib/components/dropdown/dropdown.component.mjs +396 -0
  6. package/esm2022/lib/components/global-notifications/global-notifications.component.mjs +30 -0
  7. package/esm2022/lib/components/json-editor/json-editor.component.mjs +521 -0
  8. package/esm2022/lib/components/skeleton-loader/skeleton-loader.component.mjs +33 -0
  9. package/esm2022/lib/components/toast-notification/toast-notification.component.mjs +152 -0
  10. package/esm2022/lib/elements/button/cide-ele-button.component.mjs +249 -0
  11. package/esm2022/lib/elements/file-input/file-input.component.mjs +83 -0
  12. package/esm2022/lib/elements/icon/icon.component.mjs +5 -3
  13. package/esm2022/lib/elements/input/input.component.mjs +34 -20
  14. package/esm2022/lib/elements/select/select.component.mjs +471 -0
  15. package/esm2022/lib/elements/tab/cide-ele-tab.component.mjs +74 -0
  16. package/esm2022/lib/elements/textarea/textarea.component.mjs +157 -0
  17. package/esm2022/lib/services/confirmation.service.mjs +151 -0
  18. package/esm2022/lib/services/dropdown-manager.service.mjs +93 -0
  19. package/esm2022/lib/services/notification.service.mjs +196 -0
  20. package/esm2022/lib/utils/directives/resizer/resizer.directive.mjs +231 -0
  21. package/esm2022/lib/utils/directives/tooltip/tooltip.directive.mjs +294 -0
  22. package/esm2022/lib/utils/services/elements/elements.service.mjs +9 -7
  23. package/esm2022/public-api.mjs +23 -2
  24. package/fesm2022/cloud-ide-element.mjs +4646 -47
  25. package/fesm2022/cloud-ide-element.mjs.map +1 -1
  26. package/lib/components/confirmation-modal/confirmation-modal.component.d.ts +16 -0
  27. package/lib/components/data-grid/data-grid.component.d.ts +244 -0
  28. package/lib/components/data-grid/data-grid.types.d.ts +146 -0
  29. package/lib/components/dropdown/dropdown.component.d.ts +75 -0
  30. package/lib/components/global-notifications/global-notifications.component.d.ts +5 -0
  31. package/lib/components/json-editor/json-editor.component.d.ts +116 -0
  32. package/lib/components/skeleton-loader/skeleton-loader.component.d.ts +11 -0
  33. package/lib/components/toast-notification/toast-notification.component.d.ts +13 -0
  34. package/lib/elements/button/cide-ele-button.component.d.ts +85 -0
  35. package/lib/elements/file-input/file-input.component.d.ts +25 -0
  36. package/lib/elements/input/input.component.d.ts +7 -4
  37. package/lib/elements/select/select.component.d.ts +91 -0
  38. package/lib/elements/tab/cide-ele-tab.component.d.ts +26 -0
  39. package/lib/elements/textarea/textarea.component.d.ts +47 -0
  40. package/lib/services/confirmation.service.d.ts +65 -0
  41. package/lib/services/dropdown-manager.service.d.ts +22 -0
  42. package/lib/services/notification.service.d.ts +81 -0
  43. package/lib/utils/directives/resizer/resizer.directive.d.ts +44 -0
  44. package/lib/utils/directives/tooltip/tooltip.directive.d.ts +43 -0
  45. package/package.json +32 -4
  46. package/public-api.d.ts +18 -1
  47. package/src/lib/assets/css/cide-ele-style.scss +85 -0
  48. package/src/lib/assets/css/cide-ele-variable.scss +336 -0
  49. package/esm2022/lib/elements/button/button.component.mjs +0 -60
  50. package/lib/elements/button/button.component.d.ts +0 -27
@@ -0,0 +1,74 @@
1
+ import { Component, Input, Output, EventEmitter, signal } from '@angular/core';
2
+ import { CommonModule } from '@angular/common';
3
+ import * as i0 from "@angular/core";
4
+ export class CideEleTabComponent {
5
+ constructor() {
6
+ this.tabs = [];
7
+ this.size = 'md';
8
+ this.variant = 'default';
9
+ this.fullWidth = false;
10
+ this.disabled = false;
11
+ this.tabChange = new EventEmitter();
12
+ this.activeTab = signal('');
13
+ }
14
+ ngOnInit() {
15
+ // Set initial active tab
16
+ if (this.activeTabId) {
17
+ this.activeTab.set(this.activeTabId);
18
+ }
19
+ else if (this.tabs.length > 0) {
20
+ const firstEnabledTab = this.tabs.find(tab => !tab.disabled);
21
+ if (firstEnabledTab) {
22
+ this.activeTab.set(firstEnabledTab.id);
23
+ }
24
+ }
25
+ }
26
+ onTabClick(tab) {
27
+ if (tab.disabled || this.disabled)
28
+ return;
29
+ this.activeTab.set(tab.id);
30
+ this.tabChange.emit(tab);
31
+ }
32
+ isActive(tabId) {
33
+ return this.activeTab() === tabId;
34
+ }
35
+ getTabClasses(tab) {
36
+ const baseClasses = 'cide-tab-item';
37
+ const sizeClass = `cide-tab-${this.size}`;
38
+ const variantClass = `cide-tab-${this.variant}`;
39
+ const activeClass = this.isActive(tab.id) ? 'cide-tab-active' : '';
40
+ const disabledClass = (tab.disabled || this.disabled) ? 'cide-tab-disabled' : '';
41
+ return [baseClasses, sizeClass, variantClass, activeClass, disabledClass]
42
+ .filter(Boolean)
43
+ .join(' ');
44
+ }
45
+ getContainerClasses() {
46
+ const baseClasses = 'cide-tab-container';
47
+ const variantClass = `cide-tab-container-${this.variant}`;
48
+ const fullWidthClass = this.fullWidth ? 'cide-tab-full-width' : '';
49
+ return [baseClasses, variantClass, fullWidthClass]
50
+ .filter(Boolean)
51
+ .join(' ');
52
+ }
53
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: CideEleTabComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
54
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.7", type: CideEleTabComponent, isStandalone: true, selector: "cide-ele-tab", inputs: { tabs: "tabs", activeTabId: "activeTabId", size: "size", variant: "variant", fullWidth: "fullWidth", disabled: "disabled" }, outputs: { tabChange: "tabChange" }, ngImport: i0, template: "<div [class]=\"getContainerClasses()\">\r\n @for (tab of tabs; track tab.id) {\r\n <button\r\n type=\"button\"\r\n [class]=\"getTabClasses(tab)\"\r\n [disabled]=\"tab.disabled || disabled\"\r\n (click)=\"onTabClick(tab)\"\r\n [attr.aria-selected]=\"isActive(tab.id)\"\r\n [attr.aria-controls]=\"'panel-' + tab.id\"\r\n role=\"tab\">\r\n \r\n @if (tab.icon) {\r\n <span class=\"cide-tab-icon\">{{ tab.icon }}</span>\r\n }\r\n \r\n <span class=\"cide-tab-label\">{{ tab.label }}</span>\r\n \r\n @if (tab.badge) {\r\n <span class=\"cide-tab-badge\">{{ tab.badge }}</span>\r\n }\r\n </button>\r\n }\r\n</div> ", styles: [".cide-tab-container{display:flex;border-bottom:1px solid #e5e7eb;overflow-x:auto;scrollbar-width:none}.cide-tab-container::-webkit-scrollbar{display:none}.cide-tab-container.cide-tab-full-width .cide-tab-item{flex:1}.cide-tab-container.cide-tab-container-pills{border-bottom:none;gap:.25rem;padding:.25rem;background:#f9fafb;border-radius:.375rem}.cide-tab-container.cide-tab-container-underline{border-bottom:2px solid #e5e7eb}.cide-tab-item{display:inline-flex;align-items:center;gap:.375rem;border:none;background:none;cursor:pointer;transition:all .15s ease;font-weight:500;color:#6b7280;white-space:nowrap;outline:none}.cide-tab-item:hover:not(.cide-tab-disabled){color:#374151}.cide-tab-item:focus-visible{outline:2px solid var(--cide-theme-primary-color);outline-offset:2px}.cide-tab-item.cide-tab-active{color:var(--cide-theme-primary-color)}.cide-tab-item.cide-tab-disabled{opacity:.5;cursor:not-allowed}.cide-tab-sm{padding:.375rem .75rem;font-size:.875rem}.cide-tab-sm .cide-tab-icon{font-size:.75rem}.cide-tab-sm .cide-tab-badge{font-size:.625rem;padding:.125rem .375rem}.cide-tab-md{padding:.5rem 1rem;font-size:.875rem}.cide-tab-md .cide-tab-icon{font-size:.875rem}.cide-tab-md .cide-tab-badge{font-size:.75rem;padding:.125rem .5rem}.cide-tab-lg{padding:.75rem 1.25rem;font-size:1rem}.cide-tab-lg .cide-tab-icon{font-size:1rem}.cide-tab-lg .cide-tab-badge{font-size:.75rem;padding:.25rem .5rem}.cide-tab-default{border-bottom:2px solid transparent}.cide-tab-default.cide-tab-active{border-bottom-color:var(--cide-theme-primary-color)}.cide-tab-pills{border-radius:.25rem}.cide-tab-pills:hover:not(.cide-tab-disabled){background:#f3f4f6}.cide-tab-pills.cide-tab-active{background:#fff;color:#374151;box-shadow:0 1px 2px #0000000d}.cide-tab-underline{border-bottom:2px solid transparent;position:relative}.cide-tab-underline.cide-tab-active:after{content:\"\";position:absolute;bottom:-2px;left:0;right:0;height:2px;background:var(--cide-theme-primary-color);border-radius:1px 1px 0 0}.cide-tab-icon{display:inline-flex;align-items:center;flex-shrink:0}.cide-tab-label{display:inline-flex;align-items:center}.cide-tab-badge{display:inline-flex;align-items:center;justify-content:center;background:#dc2626;color:#fff;border-radius:9999px;font-weight:600;min-width:1.25rem;height:1.25rem;line-height:1}.cide-tab-active .cide-tab-badge{background:var(--cide-theme-primary-color)}@media (max-width: 640px){.cide-tab-container{gap:0}.cide-tab-item .cide-tab-label{display:none}.cide-tab-item:not(:has(.cide-tab-icon)) .cide-tab-label{display:inline-flex}.cide-tab-sm{padding:.25rem .5rem}.cide-tab-md{padding:.375rem .75rem}.cide-tab-lg{padding:.5rem 1rem}}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }] }); }
55
+ }
56
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: CideEleTabComponent, decorators: [{
57
+ type: Component,
58
+ args: [{ selector: 'cide-ele-tab', standalone: true, imports: [CommonModule], template: "<div [class]=\"getContainerClasses()\">\r\n @for (tab of tabs; track tab.id) {\r\n <button\r\n type=\"button\"\r\n [class]=\"getTabClasses(tab)\"\r\n [disabled]=\"tab.disabled || disabled\"\r\n (click)=\"onTabClick(tab)\"\r\n [attr.aria-selected]=\"isActive(tab.id)\"\r\n [attr.aria-controls]=\"'panel-' + tab.id\"\r\n role=\"tab\">\r\n \r\n @if (tab.icon) {\r\n <span class=\"cide-tab-icon\">{{ tab.icon }}</span>\r\n }\r\n \r\n <span class=\"cide-tab-label\">{{ tab.label }}</span>\r\n \r\n @if (tab.badge) {\r\n <span class=\"cide-tab-badge\">{{ tab.badge }}</span>\r\n }\r\n </button>\r\n }\r\n</div> ", styles: [".cide-tab-container{display:flex;border-bottom:1px solid #e5e7eb;overflow-x:auto;scrollbar-width:none}.cide-tab-container::-webkit-scrollbar{display:none}.cide-tab-container.cide-tab-full-width .cide-tab-item{flex:1}.cide-tab-container.cide-tab-container-pills{border-bottom:none;gap:.25rem;padding:.25rem;background:#f9fafb;border-radius:.375rem}.cide-tab-container.cide-tab-container-underline{border-bottom:2px solid #e5e7eb}.cide-tab-item{display:inline-flex;align-items:center;gap:.375rem;border:none;background:none;cursor:pointer;transition:all .15s ease;font-weight:500;color:#6b7280;white-space:nowrap;outline:none}.cide-tab-item:hover:not(.cide-tab-disabled){color:#374151}.cide-tab-item:focus-visible{outline:2px solid var(--cide-theme-primary-color);outline-offset:2px}.cide-tab-item.cide-tab-active{color:var(--cide-theme-primary-color)}.cide-tab-item.cide-tab-disabled{opacity:.5;cursor:not-allowed}.cide-tab-sm{padding:.375rem .75rem;font-size:.875rem}.cide-tab-sm .cide-tab-icon{font-size:.75rem}.cide-tab-sm .cide-tab-badge{font-size:.625rem;padding:.125rem .375rem}.cide-tab-md{padding:.5rem 1rem;font-size:.875rem}.cide-tab-md .cide-tab-icon{font-size:.875rem}.cide-tab-md .cide-tab-badge{font-size:.75rem;padding:.125rem .5rem}.cide-tab-lg{padding:.75rem 1.25rem;font-size:1rem}.cide-tab-lg .cide-tab-icon{font-size:1rem}.cide-tab-lg .cide-tab-badge{font-size:.75rem;padding:.25rem .5rem}.cide-tab-default{border-bottom:2px solid transparent}.cide-tab-default.cide-tab-active{border-bottom-color:var(--cide-theme-primary-color)}.cide-tab-pills{border-radius:.25rem}.cide-tab-pills:hover:not(.cide-tab-disabled){background:#f3f4f6}.cide-tab-pills.cide-tab-active{background:#fff;color:#374151;box-shadow:0 1px 2px #0000000d}.cide-tab-underline{border-bottom:2px solid transparent;position:relative}.cide-tab-underline.cide-tab-active:after{content:\"\";position:absolute;bottom:-2px;left:0;right:0;height:2px;background:var(--cide-theme-primary-color);border-radius:1px 1px 0 0}.cide-tab-icon{display:inline-flex;align-items:center;flex-shrink:0}.cide-tab-label{display:inline-flex;align-items:center}.cide-tab-badge{display:inline-flex;align-items:center;justify-content:center;background:#dc2626;color:#fff;border-radius:9999px;font-weight:600;min-width:1.25rem;height:1.25rem;line-height:1}.cide-tab-active .cide-tab-badge{background:var(--cide-theme-primary-color)}@media (max-width: 640px){.cide-tab-container{gap:0}.cide-tab-item .cide-tab-label{display:none}.cide-tab-item:not(:has(.cide-tab-icon)) .cide-tab-label{display:inline-flex}.cide-tab-sm{padding:.25rem .5rem}.cide-tab-md{padding:.375rem .75rem}.cide-tab-lg{padding:.5rem 1rem}}\n"] }]
59
+ }], propDecorators: { tabs: [{
60
+ type: Input
61
+ }], activeTabId: [{
62
+ type: Input
63
+ }], size: [{
64
+ type: Input
65
+ }], variant: [{
66
+ type: Input
67
+ }], fullWidth: [{
68
+ type: Input
69
+ }], disabled: [{
70
+ type: Input
71
+ }], tabChange: [{
72
+ type: Output
73
+ }] } });
74
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2lkZS1lbGUtdGFiLmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL2Nsb3VkLWlkZS1lbGVtZW50L3NyYy9saWIvZWxlbWVudHMvdGFiL2NpZGUtZWxlLXRhYi5jb21wb25lbnQudHMiLCIuLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9jbG91ZC1pZGUtZWxlbWVudC9zcmMvbGliL2VsZW1lbnRzL3RhYi9jaWRlLWVsZS10YWIuY29tcG9uZW50Lmh0bWwiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFNBQVMsRUFBRSxLQUFLLEVBQUUsTUFBTSxFQUFFLFlBQVksRUFBRSxNQUFNLEVBQVUsTUFBTSxlQUFlLENBQUM7QUFDdkYsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLGlCQUFpQixDQUFDOztBQWlCL0MsTUFBTSxPQUFPLG1CQUFtQjtJQVBoQztRQVFXLFNBQUksR0FBYyxFQUFFLENBQUM7UUFFckIsU0FBSSxHQUF1QixJQUFJLENBQUM7UUFDaEMsWUFBTyxHQUFzQyxTQUFTLENBQUM7UUFDdkQsY0FBUyxHQUFHLEtBQUssQ0FBQztRQUNsQixhQUFRLEdBQUcsS0FBSyxDQUFDO1FBRWhCLGNBQVMsR0FBRyxJQUFJLFlBQVksRUFBVyxDQUFDO1FBRWxELGNBQVMsR0FBRyxNQUFNLENBQVMsRUFBRSxDQUFDLENBQUM7S0E4Q2hDO0lBNUNDLFFBQVE7UUFDTix5QkFBeUI7UUFDekIsSUFBSSxJQUFJLENBQUMsV0FBVyxFQUFFLENBQUM7WUFDckIsSUFBSSxDQUFDLFNBQVMsQ0FBQyxHQUFHLENBQUMsSUFBSSxDQUFDLFdBQVcsQ0FBQyxDQUFDO1FBQ3ZDLENBQUM7YUFBTSxJQUFJLElBQUksQ0FBQyxJQUFJLENBQUMsTUFBTSxHQUFHLENBQUMsRUFBRSxDQUFDO1lBQ2hDLE1BQU0sZUFBZSxHQUFHLElBQUksQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLEdBQUcsQ0FBQyxFQUFFLENBQUMsQ0FBQyxHQUFHLENBQUMsUUFBUSxDQUFDLENBQUM7WUFDN0QsSUFBSSxlQUFlLEVBQUUsQ0FBQztnQkFDcEIsSUFBSSxDQUFDLFNBQVMsQ0FBQyxHQUFHLENBQUMsZUFBZSxDQUFDLEVBQUUsQ0FBQyxDQUFDO1lBQ3pDLENBQUM7UUFDSCxDQUFDO0lBQ0gsQ0FBQztJQUVELFVBQVUsQ0FBQyxHQUFZO1FBQ3JCLElBQUksR0FBRyxDQUFDLFFBQVEsSUFBSSxJQUFJLENBQUMsUUFBUTtZQUFFLE9BQU87UUFFMUMsSUFBSSxDQUFDLFNBQVMsQ0FBQyxHQUFHLENBQUMsR0FBRyxDQUFDLEVBQUUsQ0FBQyxDQUFDO1FBQzNCLElBQUksQ0FBQyxTQUFTLENBQUMsSUFBSSxDQUFDLEdBQUcsQ0FBQyxDQUFDO0lBQzNCLENBQUM7SUFFRCxRQUFRLENBQUMsS0FBYTtRQUNwQixPQUFPLElBQUksQ0FBQyxTQUFTLEVBQUUsS0FBSyxLQUFLLENBQUM7SUFDcEMsQ0FBQztJQUVELGFBQWEsQ0FBQyxHQUFZO1FBQ3hCLE1BQU0sV0FBVyxHQUFHLGVBQWUsQ0FBQztRQUNwQyxNQUFNLFNBQVMsR0FBRyxZQUFZLElBQUksQ0FBQyxJQUFJLEVBQUUsQ0FBQztRQUMxQyxNQUFNLFlBQVksR0FBRyxZQUFZLElBQUksQ0FBQyxPQUFPLEVBQUUsQ0FBQztRQUNoRCxNQUFNLFdBQVcsR0FBRyxJQUFJLENBQUMsUUFBUSxDQUFDLEdBQUcsQ0FBQyxFQUFFLENBQUMsQ0FBQyxDQUFDLENBQUMsaUJBQWlCLENBQUMsQ0FBQyxDQUFDLEVBQUUsQ0FBQztRQUNuRSxNQUFNLGFBQWEsR0FBRyxDQUFDLEdBQUcsQ0FBQyxRQUFRLElBQUksSUFBSSxDQUFDLFFBQVEsQ0FBQyxDQUFDLENBQUMsQ0FBQyxtQkFBbUIsQ0FBQyxDQUFDLENBQUMsRUFBRSxDQUFDO1FBRWpGLE9BQU8sQ0FBQyxXQUFXLEVBQUUsU0FBUyxFQUFFLFlBQVksRUFBRSxXQUFXLEVBQUUsYUFBYSxDQUFDO2FBQ3RFLE1BQU0sQ0FBQyxPQUFPLENBQUM7YUFDZixJQUFJLENBQUMsR0FBRyxDQUFDLENBQUM7SUFDZixDQUFDO0lBRUQsbUJBQW1CO1FBQ2pCLE1BQU0sV0FBVyxHQUFHLG9CQUFvQixDQUFDO1FBQ3pDLE1BQU0sWUFBWSxHQUFHLHNCQUFzQixJQUFJLENBQUMsT0FBTyxFQUFFLENBQUM7UUFDMUQsTUFBTSxjQUFjLEdBQUcsSUFBSSxDQUFDLFNBQVMsQ0FBQyxDQUFDLENBQUMscUJBQXFCLENBQUMsQ0FBQyxDQUFDLEVBQUUsQ0FBQztRQUVuRSxPQUFPLENBQUMsV0FBVyxFQUFFLFlBQVksRUFBRSxjQUFjLENBQUM7YUFDL0MsTUFBTSxDQUFDLE9BQU8sQ0FBQzthQUNmLElBQUksQ0FBQyxHQUFHLENBQUMsQ0FBQztJQUNmLENBQUM7OEdBdkRVLG1CQUFtQjtrR0FBbkIsbUJBQW1CLG1QQ2xCaEMsZ3NCQXNCTyxtcUZEUkssWUFBWTs7MkZBSVgsbUJBQW1CO2tCQVAvQixTQUFTOytCQUNFLGNBQWMsY0FDWixJQUFJLFdBQ1AsQ0FBQyxZQUFZLENBQUM7OEJBS2QsSUFBSTtzQkFBWixLQUFLO2dCQUNHLFdBQVc7c0JBQW5CLEtBQUs7Z0JBQ0csSUFBSTtzQkFBWixLQUFLO2dCQUNHLE9BQU87c0JBQWYsS0FBSztnQkFDRyxTQUFTO3NCQUFqQixLQUFLO2dCQUNHLFFBQVE7c0JBQWhCLEtBQUs7Z0JBRUksU0FBUztzQkFBbEIsTUFBTSIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENvbXBvbmVudCwgSW5wdXQsIE91dHB1dCwgRXZlbnRFbWl0dGVyLCBzaWduYWwsIE9uSW5pdCB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xyXG5pbXBvcnQgeyBDb21tb25Nb2R1bGUgfSBmcm9tICdAYW5ndWxhci9jb21tb24nO1xyXG5cclxuZXhwb3J0IGludGVyZmFjZSBUYWJJdGVtIHtcclxuICBpZDogc3RyaW5nO1xyXG4gIGxhYmVsOiBzdHJpbmc7XHJcbiAgZGlzYWJsZWQ/OiBib29sZWFuO1xyXG4gIGljb24/OiBzdHJpbmc7XHJcbiAgYmFkZ2U/OiBzdHJpbmcgfCBudW1iZXI7XHJcbn1cclxuXHJcbkBDb21wb25lbnQoe1xyXG4gIHNlbGVjdG9yOiAnY2lkZS1lbGUtdGFiJyxcclxuICBzdGFuZGFsb25lOiB0cnVlLFxyXG4gIGltcG9ydHM6IFtDb21tb25Nb2R1bGVdLFxyXG4gIHRlbXBsYXRlVXJsOiAnLi9jaWRlLWVsZS10YWIuY29tcG9uZW50Lmh0bWwnLFxyXG4gIHN0eWxlVXJsczogWycuL2NpZGUtZWxlLXRhYi5jb21wb25lbnQuc2NzcyddXHJcbn0pXHJcbmV4cG9ydCBjbGFzcyBDaWRlRWxlVGFiQ29tcG9uZW50IGltcGxlbWVudHMgT25Jbml0IHtcclxuICBASW5wdXQoKSB0YWJzOiBUYWJJdGVtW10gPSBbXTtcclxuICBASW5wdXQoKSBhY3RpdmVUYWJJZD86IHN0cmluZztcclxuICBASW5wdXQoKSBzaXplOiAnc20nIHwgJ21kJyB8ICdsZycgPSAnbWQnO1xyXG4gIEBJbnB1dCgpIHZhcmlhbnQ6ICdkZWZhdWx0JyB8ICdwaWxscycgfCAndW5kZXJsaW5lJyA9ICdkZWZhdWx0JztcclxuICBASW5wdXQoKSBmdWxsV2lkdGggPSBmYWxzZTtcclxuICBASW5wdXQoKSBkaXNhYmxlZCA9IGZhbHNlO1xyXG5cclxuICBAT3V0cHV0KCkgdGFiQ2hhbmdlID0gbmV3IEV2ZW50RW1pdHRlcjxUYWJJdGVtPigpO1xyXG5cclxuICBhY3RpdmVUYWIgPSBzaWduYWw8c3RyaW5nPignJyk7XHJcblxyXG4gIG5nT25Jbml0KCkge1xyXG4gICAgLy8gU2V0IGluaXRpYWwgYWN0aXZlIHRhYlxyXG4gICAgaWYgKHRoaXMuYWN0aXZlVGFiSWQpIHtcclxuICAgICAgdGhpcy5hY3RpdmVUYWIuc2V0KHRoaXMuYWN0aXZlVGFiSWQpO1xyXG4gICAgfSBlbHNlIGlmICh0aGlzLnRhYnMubGVuZ3RoID4gMCkge1xyXG4gICAgICBjb25zdCBmaXJzdEVuYWJsZWRUYWIgPSB0aGlzLnRhYnMuZmluZCh0YWIgPT4gIXRhYi5kaXNhYmxlZCk7XHJcbiAgICAgIGlmIChmaXJzdEVuYWJsZWRUYWIpIHtcclxuICAgICAgICB0aGlzLmFjdGl2ZVRhYi5zZXQoZmlyc3RFbmFibGVkVGFiLmlkKTtcclxuICAgICAgfVxyXG4gICAgfVxyXG4gIH1cclxuXHJcbiAgb25UYWJDbGljayh0YWI6IFRhYkl0ZW0pIHtcclxuICAgIGlmICh0YWIuZGlzYWJsZWQgfHwgdGhpcy5kaXNhYmxlZCkgcmV0dXJuO1xyXG4gICAgXHJcbiAgICB0aGlzLmFjdGl2ZVRhYi5zZXQodGFiLmlkKTtcclxuICAgIHRoaXMudGFiQ2hhbmdlLmVtaXQodGFiKTtcclxuICB9XHJcblxyXG4gIGlzQWN0aXZlKHRhYklkOiBzdHJpbmcpOiBib29sZWFuIHtcclxuICAgIHJldHVybiB0aGlzLmFjdGl2ZVRhYigpID09PSB0YWJJZDtcclxuICB9XHJcblxyXG4gIGdldFRhYkNsYXNzZXModGFiOiBUYWJJdGVtKTogc3RyaW5nIHtcclxuICAgIGNvbnN0IGJhc2VDbGFzc2VzID0gJ2NpZGUtdGFiLWl0ZW0nO1xyXG4gICAgY29uc3Qgc2l6ZUNsYXNzID0gYGNpZGUtdGFiLSR7dGhpcy5zaXplfWA7XHJcbiAgICBjb25zdCB2YXJpYW50Q2xhc3MgPSBgY2lkZS10YWItJHt0aGlzLnZhcmlhbnR9YDtcclxuICAgIGNvbnN0IGFjdGl2ZUNsYXNzID0gdGhpcy5pc0FjdGl2ZSh0YWIuaWQpID8gJ2NpZGUtdGFiLWFjdGl2ZScgOiAnJztcclxuICAgIGNvbnN0IGRpc2FibGVkQ2xhc3MgPSAodGFiLmRpc2FibGVkIHx8IHRoaXMuZGlzYWJsZWQpID8gJ2NpZGUtdGFiLWRpc2FibGVkJyA6ICcnO1xyXG4gICAgXHJcbiAgICByZXR1cm4gW2Jhc2VDbGFzc2VzLCBzaXplQ2xhc3MsIHZhcmlhbnRDbGFzcywgYWN0aXZlQ2xhc3MsIGRpc2FibGVkQ2xhc3NdXHJcbiAgICAgIC5maWx0ZXIoQm9vbGVhbilcclxuICAgICAgLmpvaW4oJyAnKTtcclxuICB9XHJcblxyXG4gIGdldENvbnRhaW5lckNsYXNzZXMoKTogc3RyaW5nIHtcclxuICAgIGNvbnN0IGJhc2VDbGFzc2VzID0gJ2NpZGUtdGFiLWNvbnRhaW5lcic7XHJcbiAgICBjb25zdCB2YXJpYW50Q2xhc3MgPSBgY2lkZS10YWItY29udGFpbmVyLSR7dGhpcy52YXJpYW50fWA7XHJcbiAgICBjb25zdCBmdWxsV2lkdGhDbGFzcyA9IHRoaXMuZnVsbFdpZHRoID8gJ2NpZGUtdGFiLWZ1bGwtd2lkdGgnIDogJyc7XHJcbiAgICBcclxuICAgIHJldHVybiBbYmFzZUNsYXNzZXMsIHZhcmlhbnRDbGFzcywgZnVsbFdpZHRoQ2xhc3NdXHJcbiAgICAgIC5maWx0ZXIoQm9vbGVhbilcclxuICAgICAgLmpvaW4oJyAnKTtcclxuICB9XHJcbn0gIiwiPGRpdiBbY2xhc3NdPVwiZ2V0Q29udGFpbmVyQ2xhc3NlcygpXCI+XHJcbiAgQGZvciAodGFiIG9mIHRhYnM7IHRyYWNrIHRhYi5pZCkge1xyXG4gICAgPGJ1dHRvblxyXG4gICAgICB0eXBlPVwiYnV0dG9uXCJcclxuICAgICAgW2NsYXNzXT1cImdldFRhYkNsYXNzZXModGFiKVwiXHJcbiAgICAgIFtkaXNhYmxlZF09XCJ0YWIuZGlzYWJsZWQgfHwgZGlzYWJsZWRcIlxyXG4gICAgICAoY2xpY2spPVwib25UYWJDbGljayh0YWIpXCJcclxuICAgICAgW2F0dHIuYXJpYS1zZWxlY3RlZF09XCJpc0FjdGl2ZSh0YWIuaWQpXCJcclxuICAgICAgW2F0dHIuYXJpYS1jb250cm9sc109XCIncGFuZWwtJyArIHRhYi5pZFwiXHJcbiAgICAgIHJvbGU9XCJ0YWJcIj5cclxuICAgICAgXHJcbiAgICAgIEBpZiAodGFiLmljb24pIHtcclxuICAgICAgICA8c3BhbiBjbGFzcz1cImNpZGUtdGFiLWljb25cIj57eyB0YWIuaWNvbiB9fTwvc3Bhbj5cclxuICAgICAgfVxyXG4gICAgICBcclxuICAgICAgPHNwYW4gY2xhc3M9XCJjaWRlLXRhYi1sYWJlbFwiPnt7IHRhYi5sYWJlbCB9fTwvc3Bhbj5cclxuICAgICAgXHJcbiAgICAgIEBpZiAodGFiLmJhZGdlKSB7XHJcbiAgICAgICAgPHNwYW4gY2xhc3M9XCJjaWRlLXRhYi1iYWRnZVwiPnt7IHRhYi5iYWRnZSB9fTwvc3Bhbj5cclxuICAgICAgfVxyXG4gICAgPC9idXR0b24+XHJcbiAgfVxyXG48L2Rpdj4gIl19
@@ -0,0 +1,157 @@
1
+ import { CommonModule } from '@angular/common';
2
+ import { Component, EventEmitter, Input, Output, forwardRef } from '@angular/core';
3
+ import { FormsModule, NG_VALIDATORS, NG_VALUE_ACCESSOR } from '@angular/forms';
4
+ import * as i0 from "@angular/core";
5
+ import * as i1 from "@angular/common";
6
+ import * as i2 from "@angular/forms";
7
+ export class CideTextareaComponent {
8
+ constructor() {
9
+ this.label = '';
10
+ this.labelHide = false;
11
+ this.placeholder = '';
12
+ this.helperText = '';
13
+ this.errorText = '';
14
+ this.required = false;
15
+ this.disabled = false;
16
+ this.minlength = 0;
17
+ this.maxlength = 0;
18
+ this.rows = 3;
19
+ this.id = '';
20
+ this.ngModel = '';
21
+ this.size = 'sm';
22
+ this.fill = 'outline';
23
+ this.labelPlacement = 'fixed';
24
+ this.labelDir = 'start';
25
+ this.leadingIcon = '';
26
+ this.trailingIcon = '';
27
+ this.clearInput = false;
28
+ this.trailingIconInternal = '';
29
+ this.isTrailingIconAllwedClick = false;
30
+ this.ngModelChange = new EventEmitter();
31
+ this.isTouched = false;
32
+ this.isValid = true;
33
+ // eslint-disable-next-line @typescript-eslint/no-unused-vars
34
+ this.onChange = (value) => { };
35
+ this.onTouched = () => { };
36
+ this.onValidate = () => { };
37
+ }
38
+ ngOnInit() {
39
+ this.validateValue(this.ngModel);
40
+ }
41
+ ngOnChanges(changes) {
42
+ if (changes['ngModel']) {
43
+ this.validateValue(this.ngModel);
44
+ }
45
+ }
46
+ writeValue(value) {
47
+ this.ngModel = value;
48
+ this.validateValue(value);
49
+ }
50
+ registerOnChange(fn) {
51
+ this.onChange = fn;
52
+ }
53
+ registerOnTouched(fn) {
54
+ this.onTouched = fn;
55
+ }
56
+ setDisabledState(isDisabled) {
57
+ this.disabled = isDisabled;
58
+ }
59
+ validate(control) {
60
+ return this.validateValue(control.value);
61
+ }
62
+ validateValue(value) {
63
+ console.log('validateValue', value);
64
+ this.isValid = true;
65
+ return null;
66
+ }
67
+ onInput(event) {
68
+ const value = event.target.value;
69
+ this.ngModel = value;
70
+ this.onChange(value);
71
+ this.ngModelChange.emit(value);
72
+ this.validateValue(value);
73
+ }
74
+ onBlur() {
75
+ this.isTouched = true;
76
+ this.onTouched();
77
+ }
78
+ ClearInputValue() {
79
+ this.ngModel = '';
80
+ this.onChange('');
81
+ this.ngModelChange.emit('');
82
+ this.validateValue('');
83
+ }
84
+ trailingIconClick() {
85
+ // For future extensibility (e.g., password visibility)
86
+ // For now, just emit event or toggle icon if needed
87
+ }
88
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: CideTextareaComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
89
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.7", type: CideTextareaComponent, isStandalone: true, selector: "cide-ele-textarea", inputs: { label: "label", labelHide: "labelHide", placeholder: "placeholder", helperText: "helperText", errorText: "errorText", required: "required", disabled: "disabled", minlength: "minlength", maxlength: "maxlength", rows: "rows", id: "id", ngModel: "ngModel", size: "size", fill: "fill", labelPlacement: "labelPlacement", labelDir: "labelDir", leadingIcon: "leadingIcon", trailingIcon: "trailingIcon", clearInput: "clearInput" }, outputs: { ngModelChange: "ngModelChange" }, providers: [
90
+ {
91
+ provide: NG_VALUE_ACCESSOR,
92
+ useExisting: forwardRef(() => CideTextareaComponent),
93
+ multi: true
94
+ },
95
+ {
96
+ provide: NG_VALIDATORS,
97
+ multi: true,
98
+ useExisting: forwardRef(() => CideTextareaComponent),
99
+ }
100
+ ], usesOnChanges: true, ngImport: i0, template: "<div class=\"cide-textarea\" [ngClass]=\"{\r\n 'cide-element-size-xxs': (size === '2xs'),\r\n 'cide-element-size-xs': (size === 'xs'),\r\n 'cide-element-size-sm': (size === 'sm'),\r\n 'cide-element-size-md': (size === 'md'),\r\n 'cide-element-size-lg': (size === 'lg'),\r\n 'cide-element-input-label-floating': (labelPlacement === 'floating'),\r\n 'cide-element-input-label-start': (labelDir === 'start'),\r\n 'cide-element-input-label-end': (labelDir === 'end'),\r\n 'cide-element-input-label-fixed': (labelPlacement === 'fixed'),\r\n 'cide-element-input-label-less': (!label || labelHide),\r\n 'cide-element-style-outline': (fill === 'outline'),\r\n 'cide-element-style-solid': (fill === 'solid'),\r\n 'cide-element-style-standard': (fill === 'standard'),\r\n}\">\r\n <label *ngIf=\"label && !labelHide\" [for]=\"id\" class=\"cide-textarea-label\">{{ label }}</label>\r\n <div class=\"cide-element-input-wrapper\">\r\n <!-- Leading Icon -->\r\n <span class=\"cide-input-leading-icon-wrapper\" *ngIf=\"leadingIcon\">\r\n <span class=\"cide-input-leading-icon material-symbols-outlined tw-text-center\">{{leadingIcon}}</span>\r\n </span>\r\n <!-- Trailing Icon -->\r\n <span class=\"tw-absolute cide-input-trailing-icon -tw-bottom-1 tw-select-none tw-right-0\"\r\n *ngIf=\"trailingIconInternal\">\r\n <span class=\"material-symbols-outlined tw-w-8 tw-text-center !tw-text-2xl\"\r\n [ngClass]=\"{'tw-cursor-pointer': isTrailingIconAllwedClick}\" [attr.tabindex]=\"false\"\r\n (click)=\"trailingIconClick()\" (keyup)=\"trailingIconClick()\">{{trailingIconInternal}}</span>\r\n </span>\r\n <!-- Clear -->\r\n <button class=\"cide-input-clear\" *ngIf=\"clearInput && ngModel\" (click)=\"ClearInputValue()\">\r\n <span class=\"cide-input-clear-icon material-symbols-outlined\">close</span>\r\n </button>\r\n <!-- Textarea -->\r\n <textarea [id]=\"id\" [attr.placeholder]=\"placeholder\" [attr.rows]=\"rows\"\r\n [attr.minlength]=\"minlength > 0 ? minlength : null\" [attr.maxlength]=\"maxlength > 0 ? maxlength : null\"\r\n [disabled]=\"disabled\" [(ngModel)]=\"ngModel\" (input)=\"onInput($event)\" (blur)=\"onBlur()\"\r\n [ngClass]=\"[((label && labelPlacement === 'fixed') ? 'tw-rounded-e-md tw-rounded-es-md' : 'tw-rounded-md '), (!leadingIcon ? 'tw-pl-1' : ''), (trailingIconInternal ? 'tw-pr-8': ''), (!trailingIconInternal ? 'tw-pr-1' : ''), ((size === 'md') ? 'tw-h-8 tw-pt-0.5 tw-pb-0' : (size === 'sm' ? '' : '')), (labelHide ? '!tw-mt-0' : '')]\"\r\n class=\"cide-textarea-input tw-m-0 tw-w-full tw-bg-transparent tw-overflow-hidden tw-border-solid tw-py-[1px] tw-outline-none\"></textarea>\r\n </div>\r\n <span *ngIf=\"(errorText || helperText) && !isValid\" class=\"cide-textarea-help-error-text\">{{ errorText || helperText\r\n }}</span>\r\n <span *ngIf=\"helperText && isValid\" class=\"cide-textarea-help-error-text\">{{ helperText }}</span>\r\n</div>", styles: [""], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i2.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i2.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i2.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }] }); }
101
+ }
102
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: CideTextareaComponent, decorators: [{
103
+ type: Component,
104
+ args: [{ selector: 'cide-ele-textarea', standalone: true, imports: [CommonModule, FormsModule], providers: [
105
+ {
106
+ provide: NG_VALUE_ACCESSOR,
107
+ useExisting: forwardRef(() => CideTextareaComponent),
108
+ multi: true
109
+ },
110
+ {
111
+ provide: NG_VALIDATORS,
112
+ multi: true,
113
+ useExisting: forwardRef(() => CideTextareaComponent),
114
+ }
115
+ ], template: "<div class=\"cide-textarea\" [ngClass]=\"{\r\n 'cide-element-size-xxs': (size === '2xs'),\r\n 'cide-element-size-xs': (size === 'xs'),\r\n 'cide-element-size-sm': (size === 'sm'),\r\n 'cide-element-size-md': (size === 'md'),\r\n 'cide-element-size-lg': (size === 'lg'),\r\n 'cide-element-input-label-floating': (labelPlacement === 'floating'),\r\n 'cide-element-input-label-start': (labelDir === 'start'),\r\n 'cide-element-input-label-end': (labelDir === 'end'),\r\n 'cide-element-input-label-fixed': (labelPlacement === 'fixed'),\r\n 'cide-element-input-label-less': (!label || labelHide),\r\n 'cide-element-style-outline': (fill === 'outline'),\r\n 'cide-element-style-solid': (fill === 'solid'),\r\n 'cide-element-style-standard': (fill === 'standard'),\r\n}\">\r\n <label *ngIf=\"label && !labelHide\" [for]=\"id\" class=\"cide-textarea-label\">{{ label }}</label>\r\n <div class=\"cide-element-input-wrapper\">\r\n <!-- Leading Icon -->\r\n <span class=\"cide-input-leading-icon-wrapper\" *ngIf=\"leadingIcon\">\r\n <span class=\"cide-input-leading-icon material-symbols-outlined tw-text-center\">{{leadingIcon}}</span>\r\n </span>\r\n <!-- Trailing Icon -->\r\n <span class=\"tw-absolute cide-input-trailing-icon -tw-bottom-1 tw-select-none tw-right-0\"\r\n *ngIf=\"trailingIconInternal\">\r\n <span class=\"material-symbols-outlined tw-w-8 tw-text-center !tw-text-2xl\"\r\n [ngClass]=\"{'tw-cursor-pointer': isTrailingIconAllwedClick}\" [attr.tabindex]=\"false\"\r\n (click)=\"trailingIconClick()\" (keyup)=\"trailingIconClick()\">{{trailingIconInternal}}</span>\r\n </span>\r\n <!-- Clear -->\r\n <button class=\"cide-input-clear\" *ngIf=\"clearInput && ngModel\" (click)=\"ClearInputValue()\">\r\n <span class=\"cide-input-clear-icon material-symbols-outlined\">close</span>\r\n </button>\r\n <!-- Textarea -->\r\n <textarea [id]=\"id\" [attr.placeholder]=\"placeholder\" [attr.rows]=\"rows\"\r\n [attr.minlength]=\"minlength > 0 ? minlength : null\" [attr.maxlength]=\"maxlength > 0 ? maxlength : null\"\r\n [disabled]=\"disabled\" [(ngModel)]=\"ngModel\" (input)=\"onInput($event)\" (blur)=\"onBlur()\"\r\n [ngClass]=\"[((label && labelPlacement === 'fixed') ? 'tw-rounded-e-md tw-rounded-es-md' : 'tw-rounded-md '), (!leadingIcon ? 'tw-pl-1' : ''), (trailingIconInternal ? 'tw-pr-8': ''), (!trailingIconInternal ? 'tw-pr-1' : ''), ((size === 'md') ? 'tw-h-8 tw-pt-0.5 tw-pb-0' : (size === 'sm' ? '' : '')), (labelHide ? '!tw-mt-0' : '')]\"\r\n class=\"cide-textarea-input tw-m-0 tw-w-full tw-bg-transparent tw-overflow-hidden tw-border-solid tw-py-[1px] tw-outline-none\"></textarea>\r\n </div>\r\n <span *ngIf=\"(errorText || helperText) && !isValid\" class=\"cide-textarea-help-error-text\">{{ errorText || helperText\r\n }}</span>\r\n <span *ngIf=\"helperText && isValid\" class=\"cide-textarea-help-error-text\">{{ helperText }}</span>\r\n</div>" }]
116
+ }], propDecorators: { label: [{
117
+ type: Input
118
+ }], labelHide: [{
119
+ type: Input
120
+ }], placeholder: [{
121
+ type: Input
122
+ }], helperText: [{
123
+ type: Input
124
+ }], errorText: [{
125
+ type: Input
126
+ }], required: [{
127
+ type: Input
128
+ }], disabled: [{
129
+ type: Input
130
+ }], minlength: [{
131
+ type: Input
132
+ }], maxlength: [{
133
+ type: Input
134
+ }], rows: [{
135
+ type: Input
136
+ }], id: [{
137
+ type: Input
138
+ }], ngModel: [{
139
+ type: Input
140
+ }], size: [{
141
+ type: Input
142
+ }], fill: [{
143
+ type: Input
144
+ }], labelPlacement: [{
145
+ type: Input
146
+ }], labelDir: [{
147
+ type: Input
148
+ }], leadingIcon: [{
149
+ type: Input
150
+ }], trailingIcon: [{
151
+ type: Input
152
+ }], clearInput: [{
153
+ type: Input
154
+ }], ngModelChange: [{
155
+ type: Output
156
+ }] } });
157
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"textarea.component.js","sourceRoot":"","sources":["../../../../../../projects/cloud-ide-element/src/lib/elements/textarea/textarea.component.ts","../../../../../../projects/cloud-ide-element/src/lib/elements/textarea/textarea.component.html"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport { Component, EventEmitter, Input, OnChanges, OnInit, Output, SimpleChanges, forwardRef } from '@angular/core';\nimport { AbstractControl, ControlValueAccessor, FormsModule, NG_VALIDATORS, NG_VALUE_ACCESSOR, ValidationErrors, Validator } from '@angular/forms';\nimport { themeSize, elementStyleType, labelPlacementType, labelDirType } from 'cloud-ide-lms-model';\n\n@Component({\n  selector: 'cide-ele-textarea',\n  standalone: true,\n  templateUrl: './textarea.component.html',\n  styleUrl: './textarea.component.scss',\n  imports: [CommonModule, FormsModule],\n  providers: [\n    {\n      provide: NG_VALUE_ACCESSOR,\n      useExisting: forwardRef(() => CideTextareaComponent),\n      multi: true\n    },\n    {\n      provide: NG_VALIDATORS,\n      multi: true,\n      useExisting: forwardRef(() => CideTextareaComponent),\n    }\n  ]\n})\nexport class CideTextareaComponent implements ControlValueAccessor, Validator, OnInit, OnChanges {\n  @Input() label: string = '';\n  @Input() labelHide: boolean = false;\n  @Input() placeholder: string = '';\n  @Input() helperText: string = '';\n  @Input() errorText: string = '';\n  @Input() required: boolean = false;\n  @Input() disabled: boolean = false;\n  @Input() minlength: number = 0;\n  @Input() maxlength: number = 0;\n  @Input() rows: number | string = 3;\n  @Input() id: string = '';\n  @Input() ngModel: string = '';\n  @Input() size: themeSize = 'sm';\n  @Input() fill: elementStyleType = 'outline';\n  @Input() labelPlacement: labelPlacementType = 'fixed';\n  @Input() labelDir: labelDirType = 'start';\n  @Input() leadingIcon: string = '';\n  @Input() trailingIcon: string = '';\n  @Input() clearInput: boolean = false;\n  trailingIconInternal: string = '';\n  isTrailingIconAllwedClick: boolean = false;\n  @Output() ngModelChange = new EventEmitter<string>();\n\n  isTouched = false;\n  isValid = true;\n\n  // eslint-disable-next-line @typescript-eslint/no-unused-vars\n  onChange = (value: string) => {};\n  onTouched = () => {};\n  onValidate = () => {};\n\n  ngOnInit(): void {\n    this.validateValue(this.ngModel);\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes['ngModel']) {\n      this.validateValue(this.ngModel);\n    }\n  }\n\n  writeValue(value: string): void {\n    this.ngModel = value;\n    this.validateValue(value);\n  }\n\n  registerOnChange(fn: (value: string) => void): void {\n    this.onChange = fn;\n  }\n\n  registerOnTouched(fn: () => void): void {\n    this.onTouched = fn;\n  }\n\n  setDisabledState(isDisabled: boolean): void {\n    this.disabled = isDisabled;\n  }\n\n  validate(control: AbstractControl): ValidationErrors | null {\n    return this.validateValue(control.value);\n  }\n\n  validateValue(value: string): ValidationErrors | null {\n    console.log('validateValue', value);\n    this.isValid = true;\n    return null;\n  }\n\n  onInput(event: Event) {\n    const value = (event.target as HTMLTextAreaElement).value;\n    this.ngModel = value;\n    this.onChange(value);\n    this.ngModelChange.emit(value);\n    this.validateValue(value);\n  }\n\n  onBlur() {\n    this.isTouched = true;\n    this.onTouched();\n  }\n\n  ClearInputValue() {\n    this.ngModel = '';\n    this.onChange('');\n    this.ngModelChange.emit('');\n    this.validateValue('');\n  }\n\n  trailingIconClick() {\n    // For future extensibility (e.g., password visibility)\n    // For now, just emit event or toggle icon if needed\n  }\n} ","<div class=\"cide-textarea\" [ngClass]=\"{\r\n  'cide-element-size-xxs': (size === '2xs'),\r\n  'cide-element-size-xs': (size === 'xs'),\r\n  'cide-element-size-sm': (size === 'sm'),\r\n  'cide-element-size-md': (size === 'md'),\r\n  'cide-element-size-lg': (size === 'lg'),\r\n  'cide-element-input-label-floating': (labelPlacement === 'floating'),\r\n  'cide-element-input-label-start': (labelDir === 'start'),\r\n  'cide-element-input-label-end': (labelDir === 'end'),\r\n  'cide-element-input-label-fixed': (labelPlacement === 'fixed'),\r\n  'cide-element-input-label-less': (!label || labelHide),\r\n  'cide-element-style-outline': (fill === 'outline'),\r\n  'cide-element-style-solid': (fill === 'solid'),\r\n  'cide-element-style-standard': (fill === 'standard'),\r\n}\">\r\n  <label *ngIf=\"label && !labelHide\" [for]=\"id\" class=\"cide-textarea-label\">{{ label }}</label>\r\n  <div class=\"cide-element-input-wrapper\">\r\n    <!-- Leading Icon -->\r\n    <span class=\"cide-input-leading-icon-wrapper\" *ngIf=\"leadingIcon\">\r\n      <span class=\"cide-input-leading-icon material-symbols-outlined tw-text-center\">{{leadingIcon}}</span>\r\n    </span>\r\n    <!-- Trailing Icon -->\r\n    <span class=\"tw-absolute cide-input-trailing-icon -tw-bottom-1 tw-select-none tw-right-0\"\r\n      *ngIf=\"trailingIconInternal\">\r\n      <span class=\"material-symbols-outlined tw-w-8 tw-text-center !tw-text-2xl\"\r\n        [ngClass]=\"{'tw-cursor-pointer': isTrailingIconAllwedClick}\" [attr.tabindex]=\"false\"\r\n        (click)=\"trailingIconClick()\" (keyup)=\"trailingIconClick()\">{{trailingIconInternal}}</span>\r\n    </span>\r\n    <!-- Clear -->\r\n    <button class=\"cide-input-clear\" *ngIf=\"clearInput && ngModel\" (click)=\"ClearInputValue()\">\r\n      <span class=\"cide-input-clear-icon material-symbols-outlined\">close</span>\r\n    </button>\r\n    <!-- Textarea -->\r\n    <textarea [id]=\"id\" [attr.placeholder]=\"placeholder\" [attr.rows]=\"rows\"\r\n      [attr.minlength]=\"minlength > 0 ? minlength : null\" [attr.maxlength]=\"maxlength > 0 ? maxlength : null\"\r\n      [disabled]=\"disabled\" [(ngModel)]=\"ngModel\" (input)=\"onInput($event)\" (blur)=\"onBlur()\"\r\n      [ngClass]=\"[((label && labelPlacement === 'fixed') ? 'tw-rounded-e-md tw-rounded-es-md' : 'tw-rounded-md '), (!leadingIcon ? 'tw-pl-1' : ''), (trailingIconInternal ? 'tw-pr-8': ''), (!trailingIconInternal ? 'tw-pr-1' : ''), ((size === 'md') ? 'tw-h-8 tw-pt-0.5 tw-pb-0' : (size === 'sm' ? '' : '')), (labelHide ? '!tw-mt-0' : '')]\"\r\n      class=\"cide-textarea-input tw-m-0 tw-w-full tw-bg-transparent tw-overflow-hidden tw-border-solid tw-py-[1px] tw-outline-none\"></textarea>\r\n  </div>\r\n  <span *ngIf=\"(errorText || helperText) && !isValid\" class=\"cide-textarea-help-error-text\">{{ errorText || helperText\r\n    }}</span>\r\n  <span *ngIf=\"helperText && isValid\" class=\"cide-textarea-help-error-text\">{{ helperText }}</span>\r\n</div>"]}
@@ -0,0 +1,151 @@
1
+ import { Injectable, signal, computed } from '@angular/core';
2
+ import * as i0 from "@angular/core";
3
+ export class ConfirmationService {
4
+ constructor() {
5
+ this.confirmationQueue = signal([]);
6
+ this.currentRequest = signal(null);
7
+ // Computed signals
8
+ this.hasActiveConfirmation = computed(() => this.currentRequest() !== null);
9
+ this.queueLength = computed(() => this.confirmationQueue().length);
10
+ }
11
+ /**
12
+ * Ask for confirmation with a simple yes/no dialog
13
+ */
14
+ ask(options) {
15
+ return new Promise((resolve, reject) => {
16
+ const request = {
17
+ id: this.generateId(),
18
+ title: options.title,
19
+ message: options.message,
20
+ type: options.type || 'warning',
21
+ icon: options.icon || this.getDefaultIcon(options.type || 'warning'),
22
+ confirmText: options.confirmText || 'Confirm',
23
+ cancelText: options.cancelText || 'Cancel',
24
+ resolve: (value) => resolve(value),
25
+ reject
26
+ };
27
+ this.addToQueue(request);
28
+ });
29
+ }
30
+ /**
31
+ * Ask for confirmation with custom template and return custom data
32
+ */
33
+ askWithTemplate(options) {
34
+ return new Promise((resolve, reject) => {
35
+ const request = {
36
+ id: this.generateId(),
37
+ title: options.title,
38
+ message: options.message,
39
+ type: options.type || 'info',
40
+ icon: options.icon || this.getDefaultIcon(options.type || 'info'),
41
+ confirmText: options.confirmText || 'Confirm',
42
+ cancelText: options.cancelText || 'Cancel',
43
+ customTemplate: options.customTemplate,
44
+ customData: options.customData,
45
+ resolve: (value) => resolve(value),
46
+ reject
47
+ };
48
+ // Cast to unknown for queue compatibility
49
+ this.addToQueue(request);
50
+ });
51
+ }
52
+ /**
53
+ * Quick confirmation for dangerous actions
54
+ */
55
+ confirmDelete(itemName) {
56
+ return this.ask({
57
+ title: 'Confirm Delete',
58
+ message: itemName ? `Are you sure you want to delete "${itemName}"? This action cannot be undone.` : 'Are you sure you want to delete this item? This action cannot be undone.',
59
+ type: 'danger',
60
+ icon: 'delete_forever',
61
+ confirmText: 'Delete',
62
+ cancelText: 'Cancel'
63
+ });
64
+ }
65
+ /**
66
+ * Quick confirmation for permanent actions
67
+ */
68
+ confirmPermanentAction(action, itemName) {
69
+ return this.ask({
70
+ title: `Confirm ${action}`,
71
+ message: itemName ? `Are you sure you want to ${action.toLowerCase()} "${itemName}"? This action cannot be undone.` : `Are you sure you want to ${action.toLowerCase()}? This action cannot be undone.`,
72
+ type: 'danger',
73
+ icon: 'warning',
74
+ confirmText: action,
75
+ cancelText: 'Cancel'
76
+ });
77
+ }
78
+ /**
79
+ * Quick confirmation for safe actions
80
+ */
81
+ confirmSafeAction(action, itemName) {
82
+ return this.ask({
83
+ title: `Confirm ${action}`,
84
+ message: itemName ? `Are you sure you want to ${action.toLowerCase()} "${itemName}"?` : `Are you sure you want to ${action.toLowerCase()}?`,
85
+ type: 'info',
86
+ icon: 'info',
87
+ confirmText: action,
88
+ cancelText: 'Cancel'
89
+ });
90
+ }
91
+ // Internal methods
92
+ addToQueue(request) {
93
+ this.confirmationQueue.update(queue => [...queue, request]);
94
+ this.processQueue();
95
+ }
96
+ processQueue() {
97
+ if (this.currentRequest() || this.confirmationQueue().length === 0) {
98
+ return;
99
+ }
100
+ const nextRequest = this.confirmationQueue()[0];
101
+ this.currentRequest.set(nextRequest);
102
+ }
103
+ confirmCurrentRequest(data) {
104
+ const request = this.currentRequest();
105
+ if (!request)
106
+ return;
107
+ request.resolve(data !== undefined ? data : true);
108
+ this.removeCurrentRequest();
109
+ }
110
+ cancelCurrentRequest() {
111
+ const request = this.currentRequest();
112
+ if (!request)
113
+ return;
114
+ request.reject(new Error('User cancelled'));
115
+ this.removeCurrentRequest();
116
+ }
117
+ removeCurrentRequest() {
118
+ this.confirmationQueue.update(queue => queue.slice(1));
119
+ this.currentRequest.set(null);
120
+ // Process next request if any
121
+ setTimeout(() => this.processQueue(), 100);
122
+ }
123
+ generateId() {
124
+ return `confirmation-${Date.now()}-${Math.random().toString(36).substr(2, 9)}`;
125
+ }
126
+ getDefaultIcon(type) {
127
+ const iconMap = {
128
+ danger: 'error',
129
+ warning: 'warning',
130
+ info: 'info',
131
+ success: 'check_circle'
132
+ };
133
+ return iconMap[type];
134
+ }
135
+ // Public getters for components
136
+ getCurrentRequest() {
137
+ return this.currentRequest;
138
+ }
139
+ getHasActiveConfirmation() {
140
+ return this.hasActiveConfirmation;
141
+ }
142
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: ConfirmationService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
143
+ static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: ConfirmationService, providedIn: 'root' }); }
144
+ }
145
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: ConfirmationService, decorators: [{
146
+ type: Injectable,
147
+ args: [{
148
+ providedIn: 'root'
149
+ }]
150
+ }], ctorParameters: () => [] });
151
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"confirmation.service.js","sourceRoot":"","sources":["../../../../../projects/cloud-ide-element/src/lib/services/confirmation.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, signal, computed, TemplateRef } from '@angular/core';\r\n\r\nexport interface ConfirmationRequest<T = unknown> {\r\n  id: string;\r\n  title: string;\r\n  message: string;\r\n  type: 'danger' | 'warning' | 'info' | 'success';\r\n  icon?: string;\r\n  confirmText?: string;\r\n  cancelText?: string;\r\n  customTemplate?: TemplateRef<unknown>;\r\n  customData?: T;\r\n  resolve: (value: boolean | T) => void;\r\n  reject: (reason?: unknown) => void;\r\n}\r\n\r\nexport interface ConfirmationOptions<T = unknown> {\r\n  title: string;\r\n  message: string;\r\n  type?: 'danger' | 'warning' | 'info' | 'success';\r\n  icon?: string;\r\n  confirmText?: string;\r\n  cancelText?: string;\r\n  customTemplate?: TemplateRef<unknown>;\r\n  customData?: T;\r\n}\r\n\r\n@Injectable({\r\n  providedIn: 'root'\r\n})\r\nexport class ConfirmationService {\r\n  private confirmationQueue = signal<ConfirmationRequest<unknown>[]>([]);\r\n  private currentRequest = signal<ConfirmationRequest<unknown> | null>(null);\r\n  \r\n  // Computed signals\r\n  hasActiveConfirmation = computed(() => this.currentRequest() !== null);\r\n  queueLength = computed(() => this.confirmationQueue().length);\r\n\r\n  constructor() {}\r\n\r\n  /**\r\n   * Ask for confirmation with a simple yes/no dialog\r\n   */\r\n  ask(options: ConfirmationOptions<unknown>): Promise<boolean> {\r\n    return new Promise<boolean>((resolve, reject) => {\r\n      const request: ConfirmationRequest<unknown> = {\r\n        id: this.generateId(),\r\n        title: options.title,\r\n        message: options.message,\r\n        type: options.type || 'warning',\r\n        icon: options.icon || this.getDefaultIcon(options.type || 'warning'),\r\n        confirmText: options.confirmText || 'Confirm',\r\n        cancelText: options.cancelText || 'Cancel',\r\n        resolve: (value: boolean | unknown) => resolve(value as boolean),\r\n        reject\r\n      };\r\n\r\n      this.addToQueue(request);\r\n    });\r\n  }\r\n\r\n  /**\r\n   * Ask for confirmation with custom template and return custom data\r\n   */\r\n  askWithTemplate<T = unknown>(options: ConfirmationOptions<T> & { customTemplate: TemplateRef<unknown> }): Promise<T> {\r\n    return new Promise<T>((resolve, reject) => {\r\n      const request: ConfirmationRequest<T> = {\r\n        id: this.generateId(),\r\n        title: options.title,\r\n        message: options.message,\r\n        type: options.type || 'info',\r\n        icon: options.icon || this.getDefaultIcon(options.type || 'info'),\r\n        confirmText: options.confirmText || 'Confirm',\r\n        cancelText: options.cancelText || 'Cancel',\r\n        customTemplate: options.customTemplate,\r\n        customData: options.customData,\r\n        resolve: (value: boolean | T) => resolve(value as T),\r\n        reject\r\n      };\r\n\r\n      // Cast to unknown for queue compatibility\r\n      this.addToQueue(request as ConfirmationRequest<unknown>);\r\n    });\r\n  }\r\n\r\n  /**\r\n   * Quick confirmation for dangerous actions\r\n   */\r\n  confirmDelete(itemName?: string): Promise<boolean> {\r\n    return this.ask({\r\n      title: 'Confirm Delete',\r\n      message: itemName ? `Are you sure you want to delete \"${itemName}\"? This action cannot be undone.` : 'Are you sure you want to delete this item? This action cannot be undone.',\r\n      type: 'danger',\r\n      icon: 'delete_forever',\r\n      confirmText: 'Delete',\r\n      cancelText: 'Cancel'\r\n    });\r\n  }\r\n\r\n  /**\r\n   * Quick confirmation for permanent actions\r\n   */\r\n  confirmPermanentAction(action: string, itemName?: string): Promise<boolean> {\r\n    return this.ask({\r\n      title: `Confirm ${action}`,\r\n      message: itemName ? `Are you sure you want to ${action.toLowerCase()} \"${itemName}\"? This action cannot be undone.` : `Are you sure you want to ${action.toLowerCase()}? This action cannot be undone.`,\r\n      type: 'danger',\r\n      icon: 'warning',\r\n      confirmText: action,\r\n      cancelText: 'Cancel'\r\n    });\r\n  }\r\n\r\n  /**\r\n   * Quick confirmation for safe actions\r\n   */\r\n  confirmSafeAction(action: string, itemName?: string): Promise<boolean> {\r\n    return this.ask({\r\n      title: `Confirm ${action}`,\r\n      message: itemName ? `Are you sure you want to ${action.toLowerCase()} \"${itemName}\"?` : `Are you sure you want to ${action.toLowerCase()}?`,\r\n      type: 'info',\r\n      icon: 'info',\r\n      confirmText: action,\r\n      cancelText: 'Cancel'\r\n    });\r\n  }\r\n\r\n  // Internal methods\r\n  private addToQueue(request: ConfirmationRequest<unknown>): void {\r\n    this.confirmationQueue.update(queue => [...queue, request]);\r\n    this.processQueue();\r\n  }\r\n\r\n  private processQueue(): void {\r\n    if (this.currentRequest() || this.confirmationQueue().length === 0) {\r\n      return;\r\n    }\r\n\r\n    const nextRequest = this.confirmationQueue()[0];\r\n    this.currentRequest.set(nextRequest);\r\n  }\r\n\r\n  confirmCurrentRequest(data?: unknown): void {\r\n    const request = this.currentRequest();\r\n    if (!request) return;\r\n\r\n    request.resolve(data !== undefined ? data : true);\r\n    this.removeCurrentRequest();\r\n  }\r\n\r\n  cancelCurrentRequest(): void {\r\n    const request = this.currentRequest();\r\n    if (!request) return;\r\n\r\n    request.reject(new Error('User cancelled'));\r\n    this.removeCurrentRequest();\r\n  }\r\n\r\n  private removeCurrentRequest(): void {\r\n    this.confirmationQueue.update(queue => queue.slice(1));\r\n    this.currentRequest.set(null);\r\n    \r\n    // Process next request if any\r\n    setTimeout(() => this.processQueue(), 100);\r\n  }\r\n\r\n  private generateId(): string {\r\n    return `confirmation-${Date.now()}-${Math.random().toString(36).substr(2, 9)}`;\r\n  }\r\n\r\n  private getDefaultIcon(type: 'danger' | 'warning' | 'info' | 'success'): string {\r\n    const iconMap = {\r\n      danger: 'error',\r\n      warning: 'warning',\r\n      info: 'info',\r\n      success: 'check_circle'\r\n    };\r\n    return iconMap[type];\r\n  }\r\n\r\n  // Public getters for components\r\n  getCurrentRequest() {\r\n    return this.currentRequest;\r\n  }\r\n\r\n  getHasActiveConfirmation() {\r\n    return this.hasActiveConfirmation;\r\n  }\r\n}\r\n"]}
@@ -0,0 +1,93 @@
1
+ import { Injectable, signal } from '@angular/core';
2
+ import * as i0 from "@angular/core";
3
+ export class DropdownManagerService {
4
+ constructor() {
5
+ this.activeDropdowns = signal(new Map());
6
+ this.initializeGlobalListeners();
7
+ }
8
+ initializeGlobalListeners() {
9
+ // Global document click listener
10
+ this.documentClickListener = (event) => {
11
+ this.handleDocumentClick(event);
12
+ };
13
+ // Global escape key listener
14
+ this.escapeKeyListener = (event) => {
15
+ if (event.key === 'Escape') {
16
+ this.closeAllDropdowns();
17
+ }
18
+ };
19
+ // Add listeners
20
+ document.addEventListener('click', this.documentClickListener, true);
21
+ document.addEventListener('keydown', this.escapeKeyListener, true);
22
+ }
23
+ registerDropdown(dropdown) {
24
+ // Close any existing dropdowns before opening a new one
25
+ this.closeAllDropdowns();
26
+ this.activeDropdowns.update(map => {
27
+ const newMap = new Map(map);
28
+ newMap.set(dropdown.id, dropdown);
29
+ return newMap;
30
+ });
31
+ }
32
+ unregisterDropdown(id) {
33
+ this.activeDropdowns.update(map => {
34
+ const newMap = new Map(map);
35
+ newMap.delete(id);
36
+ return newMap;
37
+ });
38
+ }
39
+ closeAllDropdowns() {
40
+ const dropdowns = this.activeDropdowns();
41
+ dropdowns.forEach(dropdown => {
42
+ dropdown.close();
43
+ });
44
+ this.activeDropdowns.set(new Map());
45
+ }
46
+ closeDropdown(id) {
47
+ const dropdowns = this.activeDropdowns();
48
+ const dropdown = dropdowns.get(id);
49
+ if (dropdown) {
50
+ dropdown.close();
51
+ this.unregisterDropdown(id);
52
+ }
53
+ }
54
+ isDropdownOpen(id) {
55
+ return this.activeDropdowns().has(id);
56
+ }
57
+ handleDocumentClick(event) {
58
+ const dropdowns = this.activeDropdowns();
59
+ if (dropdowns.size === 0) {
60
+ return;
61
+ }
62
+ // Check if click is inside any open dropdown
63
+ const target = event.target;
64
+ let isClickInsideDropdown = false;
65
+ dropdowns.forEach(dropdown => {
66
+ if (dropdown.element && dropdown.element.contains(target)) {
67
+ isClickInsideDropdown = true;
68
+ }
69
+ });
70
+ // If click is outside all dropdowns, close them
71
+ if (!isClickInsideDropdown) {
72
+ this.closeAllDropdowns();
73
+ }
74
+ }
75
+ ngOnDestroy() {
76
+ // Clean up global listeners
77
+ if (this.documentClickListener) {
78
+ document.removeEventListener('click', this.documentClickListener, true);
79
+ }
80
+ if (this.escapeKeyListener) {
81
+ document.removeEventListener('keydown', this.escapeKeyListener, true);
82
+ }
83
+ }
84
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: DropdownManagerService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
85
+ static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: DropdownManagerService, providedIn: 'root' }); }
86
+ }
87
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: DropdownManagerService, decorators: [{
88
+ type: Injectable,
89
+ args: [{
90
+ providedIn: 'root'
91
+ }]
92
+ }], ctorParameters: () => [] });
93
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZHJvcGRvd24tbWFuYWdlci5zZXJ2aWNlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvY2xvdWQtaWRlLWVsZW1lbnQvc3JjL2xpYi9zZXJ2aWNlcy9kcm9wZG93bi1tYW5hZ2VyLnNlcnZpY2UudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFVBQVUsRUFBRSxNQUFNLEVBQUUsTUFBTSxlQUFlLENBQUM7O0FBV25ELE1BQU0sT0FBTyxzQkFBc0I7SUFLakM7UUFKUSxvQkFBZSxHQUFHLE1BQU0sQ0FBZ0MsSUFBSSxHQUFHLEVBQUUsQ0FBQyxDQUFDO1FBS3pFLElBQUksQ0FBQyx5QkFBeUIsRUFBRSxDQUFDO0lBQ25DLENBQUM7SUFFTyx5QkFBeUI7UUFDL0IsaUNBQWlDO1FBQ2pDLElBQUksQ0FBQyxxQkFBcUIsR0FBRyxDQUFDLEtBQVksRUFBRSxFQUFFO1lBQzVDLElBQUksQ0FBQyxtQkFBbUIsQ0FBQyxLQUFLLENBQUMsQ0FBQztRQUNsQyxDQUFDLENBQUM7UUFFRiw2QkFBNkI7UUFDN0IsSUFBSSxDQUFDLGlCQUFpQixHQUFHLENBQUMsS0FBb0IsRUFBRSxFQUFFO1lBQ2hELElBQUksS0FBSyxDQUFDLEdBQUcsS0FBSyxRQUFRLEVBQUUsQ0FBQztnQkFDM0IsSUFBSSxDQUFDLGlCQUFpQixFQUFFLENBQUM7WUFDM0IsQ0FBQztRQUNILENBQUMsQ0FBQztRQUVGLGdCQUFnQjtRQUNoQixRQUFRLENBQUMsZ0JBQWdCLENBQUMsT0FBTyxFQUFFLElBQUksQ0FBQyxxQkFBcUIsRUFBRSxJQUFJLENBQUMsQ0FBQztRQUNyRSxRQUFRLENBQUMsZ0JBQWdCLENBQUMsU0FBUyxFQUFFLElBQUksQ0FBQyxpQkFBaUIsRUFBRSxJQUFJLENBQUMsQ0FBQztJQUNyRSxDQUFDO0lBRUQsZ0JBQWdCLENBQUMsUUFBMEI7UUFDekMsd0RBQXdEO1FBQ3hELElBQUksQ0FBQyxpQkFBaUIsRUFB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QXNCLGNBRnJCLE1BQU07OzJGQUVQLHNCQUFzQjtrQkFIbEMsVUFBVTttQkFBQztvQkFDVixVQUFVLEVBQUUsTUFBTTtpQkFDbkIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBJbmplY3RhYmxlLCBzaWduYWwgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcclxuXHJcbmV4cG9ydCBpbnRlcmZhY2UgRHJvcGRvd25JbnN0YW5jZSB7XHJcbiAgaWQ6IHN0cmluZztcclxuICBjbG9zZTogKCkgPT4gdm9pZDtcclxuICBlbGVtZW50PzogRWxlbWVudDtcclxufVxyXG5cclxuQEluamVjdGFibGUoe1xyXG4gIHByb3ZpZGVkSW46ICdyb290J1xyXG59KVxyXG5leHBvcnQgY2xhc3MgRHJvcGRvd25NYW5hZ2VyU2VydmljZSB7XHJcbiAgcHJpdmF0ZSBhY3RpdmVEcm9wZG93bnMgPSBzaWduYWw8TWFwPHN0cmluZywgRHJvcGRvd25JbnN0YW5jZT4+KG5ldyBNYXAoKSk7XHJcbiAgcHJpdmF0ZSBkb2N1bWVudENsaWNrTGlzdGVuZXI/OiAoZXZlbnQ6IEV2ZW50KSA9PiB2b2lkO1xyXG4gIHByaXZhdGUgZXNjYXBlS2V5TGlzdGVuZXI/OiAoZXZlbnQ6IEtleWJvYXJkRXZlbnQpID0+IHZvaWQ7XHJcblxyXG4gIGNvbnN0cnVjdG9yKCkge1xyXG4gICAgdGhpcy5pbml0aWFsaXplR2xvYmFsTGlzdGVuZXJzKCk7XHJcbiAgfVxyXG5cclxuICBwcml2YXRlIGluaXRpYWxpemVHbG9iYWxMaXN0ZW5lcnMoKTogdm9pZCB7XHJcbiAgICAvLyBHbG9iYWwgZG9jdW1lbnQgY2xpY2sgbGlzdGVuZXJcclxuICAgIHRoaXMuZG9jdW1lbnRDbGlja0xpc3RlbmVyID0gKGV2ZW50OiBFdmVudCkgPT4ge1xyXG4gICAgICB0aGlzLmhhbmRsZURvY3VtZW50Q2xpY2soZXZlbnQpO1xyXG4gICAgfTtcclxuXHJcbiAgICAvLyBHbG9iYWwgZXNjYXBlIGtleSBsaXN0ZW5lclxyXG4gICAgdGhpcy5lc2NhcGVLZXlMaXN0ZW5lciA9IChldmVudDogS2V5Ym9hcmRFdmVudCkgPT4ge1xyXG4gICAgICBpZiAoZXZlbnQua2V5ID09PSAnRXNjYXBlJykge1xyXG4gICAgICAgIHRoaXMuY2xvc2VBbGxEcm9wZG93bnMoKTtcclxuICAgICAgfVxyXG4gICAgfTtcclxuXHJcbiAgICAvLyBBZGQgbGlzdGVuZXJzXHJcbiAgICBkb2N1bWVudC5hZGRFdmVudExpc3RlbmVyKCdjbGljaycsIHRoaXMuZG9jdW1lbnRDbGlja0xpc3RlbmVyLCB0cnVlKTtcclxuICAgIGRvY3VtZW50LmFkZEV2ZW50TGlzdGVuZXIoJ2tleWRvd24nLCB0aGlzLmVzY2FwZUtleUxpc3RlbmVyLCB0cnVlKTtcclxuICB9XHJcblxyXG4gIHJlZ2lzdGVyRHJvcGRvd24oZHJvcGRvd246IERyb3Bkb3duSW5zdGFuY2UpOiB2b2lkIHtcclxuICAgIC8vIENsb3NlIGFueSBleGlzdGluZyBkcm9wZG93bnMgYmVmb3JlIG9wZW5pbmcgYSBuZXcgb25lXHJcbiAgICB0aGlzLmNsb3NlQWxsRHJvcGRvd25zKCk7XHJcbiAgICBcclxuICAgIHRoaXMuYWN0aXZlRHJvcGRvd25zLnVwZGF0ZShtYXAgPT4ge1xyXG4gICAgICBjb25zdCBuZXdNYXAgPSBuZXcgTWFwKG1hcCk7XHJcbiAgICAgIG5ld01hcC5zZXQoZHJvcGRvd24uaWQsIGRyb3Bkb3duKTtcclxuICAgICAgcmV0dXJuIG5ld01hcDtcclxuICAgIH0pO1xyXG4gIH1cclxuXHJcbiAgdW5yZWdpc3RlckRyb3Bkb3duKGlkOiBzdHJpbmcpOiB2b2lkIHtcclxuICAgIHRoaXMuYWN0aXZlRHJvcGRvd25zLnVwZGF0ZShtYXAgPT4ge1xyXG4gICAgICBjb25zdCBuZXdNYXAgPSBuZXcgTWFwKG1hcCk7XHJcbiAgICAgIG5ld01hcC5kZWxldGUoaWQpO1xyXG4gICAgICByZXR1cm4gbmV3TWFwO1xyXG4gICAgfSk7XHJcbiAgfVxyXG5cclxuICBjbG9zZUFsbERyb3Bkb3ducygpOiB2b2lkIHtcclxuICAgIGNvbnN0IGRyb3Bkb3ducyA9IHRoaXMuYWN0aXZlRHJvcGRvd25zKCk7XHJcbiAgICBkcm9wZG93bnMuZm9yRWFjaChkcm9wZG93biA9PiB7XHJcbiAgICAgIGRyb3Bkb3duLmNsb3NlKCk7XHJcbiAgICB9KTtcclxuICAgIHRoaXMuYWN0aXZlRHJvcGRvd25zLnNldChuZXcgTWFwKCkpO1xyXG4gIH1cclxuXHJcbiAgY2xvc2VEcm9wZG93bihpZDogc3RyaW5nKTogdm9pZCB7XHJcbiAgICBjb25zdCBkcm9wZG93bnMgPSB0aGlzLmFjdGl2ZURyb3Bkb3ducygpO1xyXG4gICAgY29uc3QgZHJvcGRvd24gPSBkcm9wZG93bnMuZ2V0KGlkKTtcclxuICAgIGlmIChkcm9wZG93bikge1xyXG4gICAgICBkcm9wZG93bi5jbG9zZSgpO1xyXG4gICAgICB0aGlzLnVucmVnaXN0ZXJEcm9wZG93bihpZCk7XHJcbiAgICB9XHJcbiAgfVxyXG5cclxuICBpc0Ryb3Bkb3duT3BlbihpZDogc3RyaW5nKTogYm9vbGVhbiB7XHJcbiAgICByZXR1cm4gdGhpcy5hY3RpdmVEcm9wZG93bnMoKS5oYXMoaWQpO1xyXG4gIH1cclxuXHJcbiAgcHJpdmF0ZSBoYW5kbGVEb2N1bWVudENsaWNrKGV2ZW50OiBFdmVudCk6IHZvaWQge1xyXG4gICAgY29uc3QgZHJvcGRvd25zID0gdGhpcy5hY3RpdmVEcm9wZG93bnMoKTtcclxuICAgIFxyXG4gICAgaWYgKGRyb3Bkb3ducy5zaXplID09PSAwKSB7XHJcbiAgICAgIHJldHVybjtcclxuICAgIH1cclxuXHJcbiAgICAvLyBDaGVjayBpZiBjbGljayBpcyBpbnNpZGUgYW55IG9wZW4gZHJvcGRvd25cclxuICAgIGNvbnN0IHRhcmdldCA9IGV2ZW50LnRhcmdldCBhcyBFbGVtZW50O1xyXG4gICAgbGV0IGlzQ2xpY2tJbnNpZGVEcm9wZG93biA9IGZhbHNlO1xyXG5cclxuICAgIGRyb3Bkb3ducy5mb3JFYWNoKGRyb3Bkb3duID0+IHtcclxuICAgICAgaWYgKGRyb3Bkb3duLmVsZW1lbnQgJiYgZHJvcGRvd24uZWxlbWVudC5jb250YWlucyh0YXJnZXQpKSB7XHJcbiAgICAgICAgaXNDbGlja0luc2lkZURyb3Bkb3duID0gdHJ1ZTtcclxuICAgICAgfVxyXG4gICAgfSk7XHJcblxyXG4gICAgLy8gSWYgY2xpY2sgaXMgb3V0c2lkZSBhbGwgZHJvcGRvd25zLCBjbG9zZSB0aGVtXHJcbiAgICBpZiAoIWlzQ2xpY2tJbnNpZGVEcm9wZG93bikge1xyXG4gICAgICB0aGlzLmNsb3NlQWxsRHJvcGRvd25zKCk7XHJcbiAgICB9XHJcbiAgfVxyXG5cclxuICBuZ09uRGVzdHJveSgpOiB2b2lkIHtcclxuICAgIC8vIENsZWFuIHVwIGdsb2JhbCBsaXN0ZW5lcnNcclxuICAgIGlmICh0aGlzLmRvY3VtZW50Q2xpY2tMaXN0ZW5lcikge1xyXG4gICAgICBkb2N1bWVudC5yZW1vdmVFdmVudExpc3RlbmVyKCdjbGljaycsIHRoaXMuZG9jdW1lbnRDbGlja0xpc3RlbmVyLCB0cnVlKTtcclxuICAgIH1cclxuICAgIGlmICh0aGlzLmVzY2FwZUtleUxpc3RlbmVyKSB7XHJcbiAgICAgIGRvY3VtZW50LnJlbW92ZUV2ZW50TGlzdGVuZXIoJ2tleWRvd24nLCB0aGlzLmVzY2FwZUtleUxpc3RlbmVyLCB0cnVlKTtcclxuICAgIH1cclxuICB9XHJcbn1cclxuIl19