cloud-ide-element 0.0.1 → 1.0.2

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 +30 -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,152 @@
1
+ import { Component, inject } from '@angular/core';
2
+ import { CommonModule } from '@angular/common';
3
+ import { CideIconComponent } from '../../elements/icon/icon.component';
4
+ import { NotificationService } from '../../services/notification.service';
5
+ import * as i0 from "@angular/core";
6
+ import * as i1 from "@angular/common";
7
+ export class CideEleToastNotificationComponent {
8
+ constructor() {
9
+ this.notificationService = inject(NotificationService);
10
+ // Signals
11
+ this.activeNotifications = this.notificationService.getActiveNotifications();
12
+ }
13
+ // Methods
14
+ getToastClass(notification) {
15
+ return `toast-${notification.type}`;
16
+ }
17
+ getIconClass(type) {
18
+ return `toast-icon ${type}`;
19
+ }
20
+ removeNotification(id) {
21
+ this.notificationService.remove(id);
22
+ }
23
+ executeUndo(id) {
24
+ this.notificationService.executeUndo(id);
25
+ }
26
+ trackByNotification(index, notification) {
27
+ return notification.id;
28
+ }
29
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: CideEleToastNotificationComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
30
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.7", type: CideEleToastNotificationComponent, isStandalone: true, selector: "cide-ele-toast-notification", ngImport: i0, template: `
31
+ <!-- Toast Container -->
32
+ <div class="toast-container">
33
+ <div
34
+ *ngFor="let notification of activeNotifications(); trackBy: trackByNotification"
35
+ class="toast-notification"
36
+ [class]="getToastClass(notification)"
37
+ [@toastAnimation]>
38
+
39
+ <!-- Toast Content -->
40
+ <div class="toast-content">
41
+ <!-- Icon -->
42
+ <div class="toast-icon" [class]="getIconClass(notification.type)">
43
+ <cide-ele-icon size="sm">
44
+ {{ notification.icon }}
45
+ </cide-ele-icon>
46
+ </div>
47
+
48
+ <!-- Text Content -->
49
+ <div class="toast-text">
50
+ <div *ngIf="notification.title" class="toast-title">
51
+ {{ notification.title }}
52
+ </div>
53
+ <div class="toast-message">
54
+ {{ notification.message }}
55
+ </div>
56
+ </div>
57
+
58
+ <!-- Close Button -->
59
+ <button
60
+ type="button"
61
+ class="toast-close"
62
+ (click)="removeNotification(notification.id)"
63
+ aria-label="Close notification">
64
+ <cide-ele-icon size="xs">close</cide-ele-icon>
65
+ </button>
66
+ </div>
67
+
68
+ <!-- Undo Button -->
69
+ <div *ngIf="notification.showUndo" class="toast-actions">
70
+ <button
71
+ type="button"
72
+ class="toast-undo-btn"
73
+ (click)="executeUndo(notification.id)">
74
+ {{ notification.undoText }}
75
+ </button>
76
+ </div>
77
+
78
+ <!-- Progress Bar -->
79
+ <div
80
+ *ngIf="notification.duration > 0"
81
+ class="toast-progress"
82
+ [style.animation-duration]="notification.duration + 'ms'">
83
+ </div>
84
+ </div>
85
+ </div>
86
+ `, isInline: true, styles: [".toast-container{position:fixed;top:20px;right:20px;z-index:9990;display:flex;flex-direction:column;gap:12px;max-width:400px;width:100%}.toast-notification{background:#fff;border-radius:8px;box-shadow:0 10px 15px -3px #0000001a,0 4px 6px -2px #0000000d;border:1px solid #e5e7eb;overflow:hidden;position:relative;animation:slideInRight .3s ease-out}.toast-notification.success{border-left:4px solid #16a34a}.toast-notification.error{border-left:4px solid #dc2626}.toast-notification.warning{border-left:4px solid #d97706}.toast-notification.info{border-left:4px solid #2563eb}.toast-content{display:flex;align-items:flex-start;gap:12px;padding:16px}.toast-icon{width:24px;height:24px;border-radius:50%;display:flex;align-items:center;justify-content:center;flex-shrink:0;margin-top:2px}.toast-icon.success{background-color:#f0fdf4;color:#16a34a}.toast-icon.error{background-color:#fef2f2;color:#dc2626}.toast-icon.warning{background-color:#fffbeb;color:#d97706}.toast-icon.info{background-color:#eff6ff;color:#2563eb}.toast-text{flex:1;min-width:0}.toast-title{font-size:14px;font-weight:600;color:#111827;margin-bottom:4px;line-height:1.4}.toast-message{font-size:14px;color:#6b7280;line-height:1.4}.toast-close{background:none;border:none;color:#9ca3af;cursor:pointer;padding:4px;border-radius:4px;display:flex;align-items:center;justify-content:center;transition:all .2s ease;flex-shrink:0}.toast-close:hover{background-color:#f3f4f6;color:#6b7280}.toast-actions{padding:0 16px 16px;display:flex;justify-content:flex-end}.toast-undo-btn{background:none;border:none;color:#2563eb;font-size:14px;font-weight:500;cursor:pointer;padding:8px 12px;border-radius:6px;transition:all .2s ease}.toast-undo-btn:hover{background-color:#eff6ff}.toast-progress{position:absolute;bottom:0;left:0;height:3px;background-color:#e5e7eb;animation:progressShrink linear forwards}.toast-notification.success .toast-progress{background-color:#16a34a}.toast-notification.error .toast-progress{background-color:#dc2626}.toast-notification.warning .toast-progress{background-color:#d97706}.toast-notification.info .toast-progress{background-color:#2563eb}@keyframes slideInRight{0%{transform:translate(100%);opacity:0}to{transform:translate(0);opacity:1}}@keyframes slideOutRight{0%{transform:translate(0);opacity:1}to{transform:translate(100%);opacity:0}}@keyframes progressShrink{0%{width:100%}to{width:0%}}@media (max-width: 640px){.toast-container{top:10px;right:10px;left:10px;max-width:none}.toast-notification{border-radius:6px}.toast-content{padding:12px}.toast-actions{padding:0 12px 12px}}.toast-close:focus,.toast-undo-btn:focus{outline:2px solid #2563eb;outline-offset:2px}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: CideIconComponent, selector: "cide-ele-icon", inputs: ["size", "type", "toolTip"] }], animations: [
87
+ // Add animations here if needed
88
+ ] }); }
89
+ }
90
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: CideEleToastNotificationComponent, decorators: [{
91
+ type: Component,
92
+ args: [{ selector: 'cide-ele-toast-notification', standalone: true, imports: [CommonModule, CideIconComponent], template: `
93
+ <!-- Toast Container -->
94
+ <div class="toast-container">
95
+ <div
96
+ *ngFor="let notification of activeNotifications(); trackBy: trackByNotification"
97
+ class="toast-notification"
98
+ [class]="getToastClass(notification)"
99
+ [@toastAnimation]>
100
+
101
+ <!-- Toast Content -->
102
+ <div class="toast-content">
103
+ <!-- Icon -->
104
+ <div class="toast-icon" [class]="getIconClass(notification.type)">
105
+ <cide-ele-icon size="sm">
106
+ {{ notification.icon }}
107
+ </cide-ele-icon>
108
+ </div>
109
+
110
+ <!-- Text Content -->
111
+ <div class="toast-text">
112
+ <div *ngIf="notification.title" class="toast-title">
113
+ {{ notification.title }}
114
+ </div>
115
+ <div class="toast-message">
116
+ {{ notification.message }}
117
+ </div>
118
+ </div>
119
+
120
+ <!-- Close Button -->
121
+ <button
122
+ type="button"
123
+ class="toast-close"
124
+ (click)="removeNotification(notification.id)"
125
+ aria-label="Close notification">
126
+ <cide-ele-icon size="xs">close</cide-ele-icon>
127
+ </button>
128
+ </div>
129
+
130
+ <!-- Undo Button -->
131
+ <div *ngIf="notification.showUndo" class="toast-actions">
132
+ <button
133
+ type="button"
134
+ class="toast-undo-btn"
135
+ (click)="executeUndo(notification.id)">
136
+ {{ notification.undoText }}
137
+ </button>
138
+ </div>
139
+
140
+ <!-- Progress Bar -->
141
+ <div
142
+ *ngIf="notification.duration > 0"
143
+ class="toast-progress"
144
+ [style.animation-duration]="notification.duration + 'ms'">
145
+ </div>
146
+ </div>
147
+ </div>
148
+ `, animations: [
149
+ // Add animations here if needed
150
+ ], styles: [".toast-container{position:fixed;top:20px;right:20px;z-index:9990;display:flex;flex-direction:column;gap:12px;max-width:400px;width:100%}.toast-notification{background:#fff;border-radius:8px;box-shadow:0 10px 15px -3px #0000001a,0 4px 6px -2px #0000000d;border:1px solid #e5e7eb;overflow:hidden;position:relative;animation:slideInRight .3s ease-out}.toast-notification.success{border-left:4px solid #16a34a}.toast-notification.error{border-left:4px solid #dc2626}.toast-notification.warning{border-left:4px solid #d97706}.toast-notification.info{border-left:4px solid #2563eb}.toast-content{display:flex;align-items:flex-start;gap:12px;padding:16px}.toast-icon{width:24px;height:24px;border-radius:50%;display:flex;align-items:center;justify-content:center;flex-shrink:0;margin-top:2px}.toast-icon.success{background-color:#f0fdf4;color:#16a34a}.toast-icon.error{background-color:#fef2f2;color:#dc2626}.toast-icon.warning{background-color:#fffbeb;color:#d97706}.toast-icon.info{background-color:#eff6ff;color:#2563eb}.toast-text{flex:1;min-width:0}.toast-title{font-size:14px;font-weight:600;color:#111827;margin-bottom:4px;line-height:1.4}.toast-message{font-size:14px;color:#6b7280;line-height:1.4}.toast-close{background:none;border:none;color:#9ca3af;cursor:pointer;padding:4px;border-radius:4px;display:flex;align-items:center;justify-content:center;transition:all .2s ease;flex-shrink:0}.toast-close:hover{background-color:#f3f4f6;color:#6b7280}.toast-actions{padding:0 16px 16px;display:flex;justify-content:flex-end}.toast-undo-btn{background:none;border:none;color:#2563eb;font-size:14px;font-weight:500;cursor:pointer;padding:8px 12px;border-radius:6px;transition:all .2s ease}.toast-undo-btn:hover{background-color:#eff6ff}.toast-progress{position:absolute;bottom:0;left:0;height:3px;background-color:#e5e7eb;animation:progressShrink linear forwards}.toast-notification.success .toast-progress{background-color:#16a34a}.toast-notification.error .toast-progress{background-color:#dc2626}.toast-notification.warning .toast-progress{background-color:#d97706}.toast-notification.info .toast-progress{background-color:#2563eb}@keyframes slideInRight{0%{transform:translate(100%);opacity:0}to{transform:translate(0);opacity:1}}@keyframes slideOutRight{0%{transform:translate(0);opacity:1}to{transform:translate(100%);opacity:0}}@keyframes progressShrink{0%{width:100%}to{width:0%}}@media (max-width: 640px){.toast-container{top:10px;right:10px;left:10px;max-width:none}.toast-notification{border-radius:6px}.toast-content{padding:12px}.toast-actions{padding:0 12px 12px}}.toast-close:focus,.toast-undo-btn:focus{outline:2px solid #2563eb;outline-offset:2px}\n"] }]
151
+ }] });
152
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"toast-notification.component.js","sourceRoot":"","sources":["../../../../../../projects/cloud-ide-element/src/lib/components/toast-notification/toast-notification.component.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAoB,MAAM,EAAE,MAAM,eAAe,CAAC;AACpE,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;AAC/C,OAAO,EAAE,iBAAiB,EAAE,MAAM,oCAAoC,CAAC;AACvE,OAAO,EAAE,mBAAmB,EAAoB,MAAM,qCAAqC,CAAC;;;AAkS5F,MAAM,OAAO,iCAAiC;IAhS9C;QAiSU,wBAAmB,GAAG,MAAM,CAAC,mBAAmB,CAAC,CAAC;QAE1D,UAAU;QACV,wBAAmB,GAAG,IAAI,CAAC,mBAAmB,CAAC,sBAAsB,EAAE,CAAC;KAsBzE;IApBC,UAAU;IACV,aAAa,CAAC,YAA8B;QAC1C,OAAO,SAAS,YAAY,CAAC,IAAI,EAAE,CAAC;IACtC,CAAC;IAED,YAAY,CAAC,IAAY;QACvB,OAAO,cAAc,IAAI,EAAE,CAAC;IAC9B,CAAC;IAED,kBAAkB,CAAC,EAAU;QAC3B,IAAI,CAAC,mBAAmB,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC;IACtC,CAAC;IAED,WAAW,CAAC,EAAU;QACpB,IAAI,CAAC,mBAAmB,CAAC,WAAW,CAAC,EAAE,CAAC,CAAC;IAC3C,CAAC;IAED,mBAAmB,CAAC,KAAa,EAAE,YAA8B;QAC/D,OAAO,YAAY,CAAC,EAAE,CAAC;IACzB,CAAC;8GAzBU,iCAAiC;kGAAjC,iCAAiC,uFA5RlC;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAwDT,grFAzDS,YAAY,gQAAE,iBAAiB,iFAyR7B;QACV,gCAAgC;SACjC;;2FAEU,iCAAiC;kBAhS7C,SAAS;+BACE,6BAA6B,cAC3B,IAAI,WACP,CAAC,YAAY,EAAE,iBAAiB,CAAC,YAChC;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAwDT,cAgOW;oBACV,gCAAgC;qBACjC","sourcesContent":["import { Component, signal, computed, inject } from '@angular/core';\nimport { CommonModule } from '@angular/common';\nimport { CideIconComponent } from '../../elements/icon/icon.component';\nimport { NotificationService, NotificationItem } from '../../services/notification.service';\n\n@Component({\n  selector: 'cide-ele-toast-notification',\n  standalone: true,\n  imports: [CommonModule, CideIconComponent],\n  template: `\n    <!-- Toast Container -->\n    <div class=\"toast-container\">\n      <div \n        *ngFor=\"let notification of activeNotifications(); trackBy: trackByNotification\"\n        class=\"toast-notification\"\n        [class]=\"getToastClass(notification)\"\n        [@toastAnimation]>\n        \n        <!-- Toast Content -->\n        <div class=\"toast-content\">\n          <!-- Icon -->\n          <div class=\"toast-icon\" [class]=\"getIconClass(notification.type)\">\n            <cide-ele-icon size=\"sm\">\n              {{ notification.icon }}\n            </cide-ele-icon>\n          </div>\n          \n          <!-- Text Content -->\n          <div class=\"toast-text\">\n            <div *ngIf=\"notification.title\" class=\"toast-title\">\n              {{ notification.title }}\n            </div>\n            <div class=\"toast-message\">\n              {{ notification.message }}\n            </div>\n          </div>\n          \n          <!-- Close Button -->\n          <button \n            type=\"button\"\n            class=\"toast-close\"\n            (click)=\"removeNotification(notification.id)\"\n            aria-label=\"Close notification\">\n            <cide-ele-icon size=\"xs\">close</cide-ele-icon>\n          </button>\n        </div>\n        \n        <!-- Undo Button -->\n        <div *ngIf=\"notification.showUndo\" class=\"toast-actions\">\n          <button \n            type=\"button\"\n            class=\"toast-undo-btn\"\n            (click)=\"executeUndo(notification.id)\">\n            {{ notification.undoText }}\n          </button>\n        </div>\n        \n        <!-- Progress Bar -->\n        <div \n          *ngIf=\"notification.duration > 0\"\n          class=\"toast-progress\"\n          [style.animation-duration]=\"notification.duration + 'ms'\">\n        </div>\n      </div>\n    </div>\n  `,\n  styles: [`\n    .toast-container {\n      position: fixed;\n      top: 20px;\n      right: 20px;\n      z-index: 9990;\n      display: flex;\n      flex-direction: column;\n      gap: 12px;\n      max-width: 400px;\n      width: 100%;\n    }\n\n    .toast-notification {\n      background: white;\n      border-radius: 8px;\n      box-shadow: 0 10px 15px -3px rgba(0, 0, 0, 0.1), 0 4px 6px -2px rgba(0, 0, 0, 0.05);\n      border: 1px solid #e5e7eb;\n      overflow: hidden;\n      position: relative;\n      animation: slideInRight 0.3s ease-out;\n    }\n\n    .toast-notification.success {\n      border-left: 4px solid #16a34a;\n    }\n\n    .toast-notification.error {\n      border-left: 4px solid #dc2626;\n    }\n\n    .toast-notification.warning {\n      border-left: 4px solid #d97706;\n    }\n\n    .toast-notification.info {\n      border-left: 4px solid #2563eb;\n    }\n\n    .toast-content {\n      display: flex;\n      align-items: flex-start;\n      gap: 12px;\n      padding: 16px;\n    }\n\n    .toast-icon {\n      width: 24px;\n      height: 24px;\n      border-radius: 50%;\n      display: flex;\n      align-items: center;\n      justify-content: center;\n      flex-shrink: 0;\n      margin-top: 2px;\n    }\n\n    .toast-icon.success {\n      background-color: #f0fdf4;\n      color: #16a34a;\n    }\n\n    .toast-icon.error {\n      background-color: #fef2f2;\n      color: #dc2626;\n    }\n\n    .toast-icon.warning {\n      background-color: #fffbeb;\n      color: #d97706;\n    }\n\n    .toast-icon.info {\n      background-color: #eff6ff;\n      color: #2563eb;\n    }\n\n    .toast-text {\n      flex: 1;\n      min-width: 0;\n    }\n\n    .toast-title {\n      font-size: 14px;\n      font-weight: 600;\n      color: #111827;\n      margin-bottom: 4px;\n      line-height: 1.4;\n    }\n\n    .toast-message {\n      font-size: 14px;\n      color: #6b7280;\n      line-height: 1.4;\n    }\n\n    .toast-close {\n      background: none;\n      border: none;\n      color: #9ca3af;\n      cursor: pointer;\n      padding: 4px;\n      border-radius: 4px;\n      display: flex;\n      align-items: center;\n      justify-content: center;\n      transition: all 0.2s ease;\n      flex-shrink: 0;\n    }\n\n    .toast-close:hover {\n      background-color: #f3f4f6;\n      color: #6b7280;\n    }\n\n    .toast-actions {\n      padding: 0 16px 16px;\n      display: flex;\n      justify-content: flex-end;\n    }\n\n    .toast-undo-btn {\n      background: none;\n      border: none;\n      color: #2563eb;\n      font-size: 14px;\n      font-weight: 500;\n      cursor: pointer;\n      padding: 8px 12px;\n      border-radius: 6px;\n      transition: all 0.2s ease;\n    }\n\n    .toast-undo-btn:hover {\n      background-color: #eff6ff;\n    }\n\n    .toast-progress {\n      position: absolute;\n      bottom: 0;\n      left: 0;\n      height: 3px;\n      background-color: #e5e7eb;\n      animation: progressShrink linear forwards;\n    }\n\n    .toast-notification.success .toast-progress {\n      background-color: #16a34a;\n    }\n\n    .toast-notification.error .toast-progress {\n      background-color: #dc2626;\n    }\n\n    .toast-notification.warning .toast-progress {\n      background-color: #d97706;\n    }\n\n    .toast-notification.info .toast-progress {\n      background-color: #2563eb;\n    }\n\n    /* Animations */\n    @keyframes slideInRight {\n      from {\n        transform: translateX(100%);\n        opacity: 0;\n      }\n      to {\n        transform: translateX(0);\n        opacity: 1;\n      }\n    }\n\n    @keyframes slideOutRight {\n      from {\n        transform: translateX(0);\n        opacity: 1;\n      }\n      to {\n        transform: translateX(100%);\n        opacity: 0;\n      }\n    }\n\n    @keyframes progressShrink {\n      from {\n        width: 100%;\n      }\n      to {\n        width: 0%;\n      }\n    }\n\n    /* Responsive Design */\n    @media (max-width: 640px) {\n      .toast-container {\n        top: 10px;\n        right: 10px;\n        left: 10px;\n        max-width: none;\n      }\n\n      .toast-notification {\n        border-radius: 6px;\n      }\n\n      .toast-content {\n        padding: 12px;\n      }\n\n      .toast-actions {\n        padding: 0 12px 12px;\n      }\n    }\n\n    /* Focus styles for accessibility */\n    .toast-close:focus,\n    .toast-undo-btn:focus {\n      outline: 2px solid #2563eb;\n      outline-offset: 2px;\n    }\n  `],\n  animations: [\n    // Add animations here if needed\n  ]\n})\nexport class CideEleToastNotificationComponent {\n  private notificationService = inject(NotificationService);\n\n  // Signals\n  activeNotifications = this.notificationService.getActiveNotifications();\n\n  // Methods\n  getToastClass(notification: NotificationItem): string {\n    return `toast-${notification.type}`;\n  }\n\n  getIconClass(type: string): string {\n    return `toast-icon ${type}`;\n  }\n\n  removeNotification(id: string): void {\n    this.notificationService.remove(id);\n  }\n\n  executeUndo(id: string): void {\n    this.notificationService.executeUndo(id);\n  }\n\n  trackByNotification(index: number, notification: NotificationItem): string {\n    return notification.id;\n  }\n}\n"]}
@@ -0,0 +1,249 @@
1
+ import { Component, EventEmitter, HostListener, Input, Output } from '@angular/core';
2
+ import { CommonModule } from '@angular/common';
3
+ import { CideSpinnerComponent } from '../spinner/spinner.component';
4
+ import { Subject, debounceTime, takeUntil } from 'rxjs';
5
+ import * as i0 from "@angular/core";
6
+ import * as i1 from "../../../public-api";
7
+ import * as i2 from "@angular/router";
8
+ import * as i3 from "@angular/common";
9
+ /**
10
+ * Advanced Angular button component that provides a comprehensive button implementation
11
+ * with loading state, styling variants, animations, tooltips, and other features.
12
+ *
13
+ * This component uses proper Angular templating to avoid hydration issues (NG0500).
14
+ */
15
+ export class CideEleButtonComponent {
16
+ constructor(elementService, router, renderer, elementRef) {
17
+ this.elementService = elementService;
18
+ this.router = router;
19
+ this.renderer = renderer;
20
+ this.elementRef = elementRef;
21
+ this.destroy$ = new Subject();
22
+ this.clickCount = 0;
23
+ this.clickDebouncer$ = new Subject();
24
+ // INPUTS
25
+ /**
26
+ * @description Label of button like Submit, Update, Add, etc.
27
+ * Note: it is only used in case you did not set label between tags (via ng-content)
28
+ */
29
+ this.label = "";
30
+ /** @description Button style variant */
31
+ this.variant = 'primary';
32
+ /** @description Button size */
33
+ this.size = 'md';
34
+ /** @description Button HTML type attribute */
35
+ this.type = 'button';
36
+ /** @description Button shape */
37
+ this.shape = 'default';
38
+ /** @description Button shadow elevation */
39
+ this.elevation = 'medium';
40
+ /** @description Button disabled state */
41
+ this.disabled = false;
42
+ /** @description Unique ID of control, used to differentiate elements */
43
+ this.id = '';
44
+ /** @description Enables loading state with spinner */
45
+ this.loading = false;
46
+ /** @description Makes button take full width of container */
47
+ this.fullWidth = false;
48
+ /** @description Icon to display on the left side of the button */
49
+ this.leftIcon = '';
50
+ /** @description Icon to display on the right side of the button */
51
+ this.rightIcon = '';
52
+ /** @description Additional CSS classes to apply */
53
+ this.customClass = '';
54
+ /** @description Tooltip text to display on hover */
55
+ this.tooltip = '';
56
+ /** @description Aria label for accessibility */
57
+ this.ariaLabel = '';
58
+ /** @description Test ID for automated testing */
59
+ this.testId = '';
60
+ /** @description Route to navigate to when clicked (for link buttons) */
61
+ this.routerLink = '';
62
+ /** @description Router navigation extras */
63
+ this.routerExtras = {};
64
+ /** @description Prevent multiple rapid clicks (debounce) */
65
+ this.preventDoubleClick = true;
66
+ /** @description Animation on click */
67
+ this.animated = true;
68
+ // OUTPUTS
69
+ /**
70
+ * @description Click event of button
71
+ */
72
+ this.btnClick = new EventEmitter();
73
+ /**
74
+ * @description Double click event
75
+ */
76
+ this.doubleClick = new EventEmitter();
77
+ // Setup debounced click handler to prevent double-clicks
78
+ this.clickDebouncer$
79
+ .pipe(debounceTime(300), takeUntil(this.destroy$))
80
+ .subscribe({
81
+ next: (event) => {
82
+ this.handleClick(event);
83
+ }
84
+ });
85
+ }
86
+ onClick(event) {
87
+ if (this.disabled || this.loading) {
88
+ event.preventDefault();
89
+ event.stopPropagation();
90
+ return;
91
+ }
92
+ if (this.animated) {
93
+ this.addClickAnimation();
94
+ }
95
+ if (this.preventDoubleClick) {
96
+ this.clickDebouncer$.next(event);
97
+ }
98
+ else {
99
+ this.handleClick(event);
100
+ }
101
+ }
102
+ handleClick(event) {
103
+ // Handle navigation for routerLink
104
+ if (this.routerLink && this.routerLink.length) {
105
+ event.preventDefault();
106
+ this.router.navigate(Array.isArray(this.routerLink) ? this.routerLink : [this.routerLink], this.routerExtras);
107
+ }
108
+ // Handle double-click detection
109
+ this.clickCount++;
110
+ setTimeout(() => {
111
+ if (this.clickCount === 2) {
112
+ this.doubleClick.emit(event);
113
+ }
114
+ this.clickCount = 0;
115
+ }, 300);
116
+ // Emit standard click event
117
+ this.btnClick.emit(event);
118
+ }
119
+ addClickAnimation() {
120
+ // Add ripple effect
121
+ const ripple = this.renderer.createElement('span');
122
+ this.renderer.addClass(ripple, 'ripple-effect');
123
+ // Position the ripple at click position
124
+ const rect = this.elementRef.nativeElement.getBoundingClientRect();
125
+ const size = Math.max(rect.width, rect.height);
126
+ // Set ripple style
127
+ this.renderer.setStyle(ripple, 'width', `${size}px`);
128
+ this.renderer.setStyle(ripple, 'height', `${size}px`);
129
+ this.renderer.setStyle(ripple, 'left', `${0}px`);
130
+ this.renderer.setStyle(ripple, 'top', `${0}px`);
131
+ // Add ripple to button
132
+ this.renderer.appendChild(this.elementRef.nativeElement, ripple);
133
+ // Modern ES2022+ pattern: Use Promise-based delay
134
+ const delay = (ms) => new Promise(resolve => setTimeout(resolve, ms));
135
+ delay(600).then(() => {
136
+ this.renderer.removeChild(this.elementRef.nativeElement, ripple);
137
+ });
138
+ }
139
+ ngOnInit() {
140
+ if (this.id) {
141
+ this.getControlData();
142
+ }
143
+ // Set aria-label if not explicitly provided
144
+ if (!this.ariaLabel && this.label) {
145
+ this.ariaLabel = this.label;
146
+ }
147
+ }
148
+ ngOnChanges(changes) {
149
+ // Update aria-label when label changes
150
+ if (changes['label'] && !this.ariaLabel) {
151
+ this.ariaLabel = changes['label'].currentValue;
152
+ }
153
+ }
154
+ ngOnDestroy() {
155
+ this.destroy$.next();
156
+ this.destroy$.complete();
157
+ }
158
+ async getControlData() {
159
+ const cide_element_data = await this.elementService?.getElementData({ sype_key: this.id });
160
+ if (cide_element_data) {
161
+ this.label = cide_element_data?.sype_label;
162
+ }
163
+ }
164
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: CideEleButtonComponent, deps: [{ token: i1.CideElementsService }, { token: i2.Router }, { token: i0.Renderer2 }, { token: i0.ElementRef }], target: i0.ɵɵFactoryTarget.Component }); }
165
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.7", type: CideEleButtonComponent, isStandalone: true, selector: "button[cideEleButton], a[cideEleButton]", inputs: { label: "label", variant: "variant", size: "size", type: "type", shape: "shape", elevation: "elevation", disabled: "disabled", id: "id", loading: "loading", fullWidth: "fullWidth", leftIcon: "leftIcon", rightIcon: "rightIcon", customClass: "customClass", tooltip: "tooltip", ariaLabel: "ariaLabel", testId: "testId", routerLink: "routerLink", routerExtras: "routerExtras", preventDoubleClick: "preventDoubleClick", animated: "animated" }, outputs: { btnClick: "btnClick", doubleClick: "doubleClick" }, host: { listeners: { "click": "onClick($event)" }, properties: { "class.cide-button-disabled": "disabled || loading", "attr.disabled": "(disabled || loading) ? true : null", "class.primary": "variant === \"primary\"", "class.secondary": "variant === \"secondary\"", "class.outline": "variant === \"outline\"", "class.text": "variant === \"text\"", "class.link": "variant === \"link\"", "class.success": "variant === \"success\"", "class.danger": "variant === \"danger\"", "class.warning": "variant === \"warning\"", "class.info": "variant === \"info\"", "class.xs": "size === \"xs\"", "class.sm": "size === \"sm\"", "class.md": "size === \"md\"", "class.lg": "size === \"lg\"", "class.xl": "size === \"xl\"", "class.rounded": "shape === \"rounded\"", "class.pill": "shape === \"pill\"", "class.square": "shape === \"square\"", "class.elevation-none": "elevation === \"none\"", "class.elevation-low": "elevation === \"low\"", "class.elevation-medium": "elevation === \"medium\"", "class.elevation-high": "elevation === \"high\"", "attr.type": "type", "attr.aria-label": "ariaLabel", "attr.aria-disabled": "disabled || loading", "attr.data-testid": "testId", "attr.title": "tooltip", "class": "customClass" }, classAttribute: "cide-button tw-rounded-md tw-text-white tw-py-0.5 tw-select-none tw-justify-around tw-flex" }, usesOnChanges: true, ngImport: i0, template: "<!-- Button content container -->\r\n<div class=\"tw-flex tw-items-center tw-justify-center cide-ele-btn-content tw-w-full\">\r\n <!-- Left icon -->\r\n @if (leftIcon) {\r\n <span class=\"tw-icon-container tw-mr-2\">\r\n <i class=\"tw-text-base material-symbols-outlined\">{{leftIcon}}</i>\r\n </span>\r\n }\r\n \r\n <!-- Loading spinner -->\r\n @if (loading) {\r\n <cide-ele-spinner\r\n class=\"tw-inline-block tw-my-1 tw-mr-2\" \r\n size=\"xs\"\r\n [ngClass]=\"{'cide-pill-disabled': (disabled || loading)}\">\r\n </cide-ele-spinner>\r\n }\r\n \r\n <!-- Button label or content -->\r\n @if (label) {\r\n <span>{{ label }}</span>\r\n } @else {\r\n <span class=\"tw-flex tw-items-center tw-justify-center\">\r\n <ng-content></ng-content>\r\n </span>\r\n }\r\n \r\n <!-- Right icon -->\r\n @if (rightIcon) {\r\n <span class=\"tw-icon-container tw-ml-2\">\r\n <i class=\"tw-text-base material-symbols-outlined\">{{rightIcon}}</i>\r\n </span>\r\n }\r\n \r\n <!-- Spacer for spinner when loading to maintain button width -->\r\n @if (loading) {\r\n <span class=\"tw-w-6 tw-p-1 tw-mr-2\"></span>\r\n }\r\n</div>\r\n", styles: [":host{position:relative;background-color:var(--cide-theme-color-brand-primary, #3b82f6);cursor:pointer;font-family:inherit;font-weight:500;outline:none;overflow:hidden;transition:all .2s ease-in-out;border:none;text-align:center;vertical-align:middle;text-decoration:none;line-height:1.5;letter-spacing:.025em;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;-webkit-appearance:none;-moz-appearance:none;appearance:none}:host:focus-visible{outline:2px solid var(--cide-theme-color-brand-primary);outline-offset:2px;box-shadow:0 0 0 3px #3b82f64d}:host:hover:not(:disabled){background-color:var(--cide-theme-color-brand-primary-hover, #2563eb);transform:translateY(-1px)}:host:active:not(:disabled){transform:translateY(1px)}:host.cide-button-disabled{background-color:var(--cide-button-background-disabled, #9ca3af)!important;cursor:var(--cide-button-cursor-disabled, not-allowed)!important;opacity:.7;pointer-events:none;transform:none!important}:host.xs{font-size:.75rem;padding:.2rem .4rem}:host.sm{font-size:.875rem;padding:.25rem .5rem}:host.md{font-size:1rem;padding:.25rem .75rem}:host.lg{font-size:1.125rem;padding:.75rem 1rem}:host.xl{font-size:1.25rem;padding:1rem 1.5rem}:host.rounded{border-radius:.5rem}:host.pill{border-radius:9999px}:host.square{border-radius:0}:host.elevation-none{box-shadow:none}:host.elevation-low{box-shadow:0 1px 2px #0000000d}:host.elevation-medium{box-shadow:0 4px 6px -1px #0000001a,0 2px 4px -1px #0000000f}:host.elevation-high{box-shadow:0 10px 15px -3px #0000001a,0 4px 6px -2px #0000000d}.ripple-effect{position:absolute;border-radius:50%;background-color:#fff6;transform:scale(0);animation:ripple .6s linear;pointer-events:none}@keyframes ripple{to{transform:scale(4);opacity:0}}:host{color:#fff}:host.secondary{background-color:var(--cide-theme-secondary-color, #4b5563);color:#fff}:host.secondary:hover:not(:disabled){background-color:var(--cide-theme-secondary-color-hover, #374151)}:host.success{background-color:var(--cide-theme-success-color, #10b981);color:#fff}:host.success:hover:not(:disabled){background-color:var(--cide-theme-success-color-hover, #059669)}:host.danger{background-color:var(--cide-theme-danger-color, #ef4444);color:#fff}:host.danger:hover:not(:disabled){background-color:var(--cide-theme-danger-color-hover, #dc2626)}:host.warning{background-color:var(--cide-theme-warning-color, #f59e0b);color:#fff}:host.warning:hover:not(:disabled){background-color:var(--cide-theme-warning-color-hover, #d97706)}:host.info{background-color:var(--cide-theme-info-color, #3b82f6);color:#fff}:host.info:hover:not(:disabled){background-color:var(--cide-theme-info-color-hover, #2563eb)}:host.outline{background-color:transparent;color:var(--cide-theme-color-brand-primary, #3b82f6);border:1px solid var(--cide-theme-color-brand-primary, #3b82f6)}:host.outline:hover:not(:disabled){background-color:#3b82f60d}:host.outline.secondary{color:var(--cide-theme-secondary-color, #4b5563);border-color:var(--cide-theme-secondary-color, #4b5563)}:host.outline.success{color:var(--cide-theme-success-color, #10b981);border-color:var(--cide-theme-success-color, #10b981)}:host.outline.danger{color:var(--cide-theme-danger-color, #ef4444);border-color:var(--cide-theme-danger-color, #ef4444)}:host.outline.warning{color:var(--cide-theme-warning-color, #f59e0b);border-color:var(--cide-theme-warning-color, #f59e0b)}:host.outline.info{color:var(--cide-theme-info-color, #3b82f6);border-color:var(--cide-theme-info-color, #3b82f6)}:host.text{background-color:transparent;color:var(--cide-theme-color-brand-primary, #3b82f6);border:none;padding-left:.5rem;padding-right:.5rem}:host.text:hover:not(:disabled){background-color:#3b82f60d;text-decoration:underline}:host.text.secondary{color:var(--cide-theme-secondary-color, #4b5563)}:host.text.success{color:var(--cide-theme-success-color, #10b981)}:host.text.danger{color:var(--cide-theme-danger-color, #ef4444)}:host.text.warning{color:var(--cide-theme-warning-color, #f59e0b)}:host.text.info{color:var(--cide-theme-info-color, #3b82f6)}:host.link{background-color:transparent;color:var(--cide-theme-color-brand-primary, #3b82f6);border:none;text-decoration:underline;padding:0;font-weight:400}:host.link:hover:not(:disabled){color:var(--cide-theme-color-brand-primary-hover, #2563eb);text-decoration:underline}:host.link.secondary{color:var(--cide-theme-secondary-color, #4b5563)}:host.link.success{color:var(--cide-theme-success-color, #10b981)}:host.link.danger{color:var(--cide-theme-danger-color, #ef4444)}:host.link.warning{color:var(--cide-theme-warning-color, #f59e0b)}:host.link.info{color:var(--cide-theme-info-color, #3b82f6)}.tw-icon-container{display:inline-flex;align-items:center;justify-content:center}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i3.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "component", type: CideSpinnerComponent, selector: "cide-ele-spinner", inputs: ["size", "type"] }] }); }
166
+ }
167
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: CideEleButtonComponent, decorators: [{
168
+ type: Component,
169
+ args: [{ selector: 'button[cideEleButton], a[cideEleButton]', standalone: true, imports: [CommonModule, CideSpinnerComponent], host: {
170
+ 'class': 'cide-button tw-rounded-md tw-text-white tw-py-0.5 tw-select-none tw-justify-around tw-flex',
171
+ '[class.cide-button-disabled]': 'disabled || loading',
172
+ '[attr.disabled]': '(disabled || loading) ? true : null',
173
+ '[class.primary]': 'variant === "primary"',
174
+ '[class.secondary]': 'variant === "secondary"',
175
+ '[class.outline]': 'variant === "outline"',
176
+ '[class.text]': 'variant === "text"',
177
+ '[class.link]': 'variant === "link"',
178
+ '[class.success]': 'variant === "success"',
179
+ '[class.danger]': 'variant === "danger"',
180
+ '[class.warning]': 'variant === "warning"',
181
+ '[class.info]': 'variant === "info"',
182
+ '[class.xs]': 'size === "xs"',
183
+ '[class.sm]': 'size === "sm"',
184
+ '[class.md]': 'size === "md"',
185
+ '[class.lg]': 'size === "lg"',
186
+ '[class.xl]': 'size === "xl"',
187
+ '[class.rounded]': 'shape === "rounded"',
188
+ '[class.pill]': 'shape === "pill"',
189
+ '[class.square]': 'shape === "square"',
190
+ '[class.elevation-none]': 'elevation === "none"',
191
+ '[class.elevation-low]': 'elevation === "low"',
192
+ '[class.elevation-medium]': 'elevation === "medium"',
193
+ '[class.elevation-high]': 'elevation === "high"',
194
+ '[attr.type]': 'type',
195
+ '[attr.aria-label]': 'ariaLabel',
196
+ '[attr.aria-disabled]': 'disabled || loading',
197
+ '[attr.data-testid]': 'testId',
198
+ '[attr.title]': 'tooltip',
199
+ '[class]': 'customClass'
200
+ }, template: "<!-- Button content container -->\r\n<div class=\"tw-flex tw-items-center tw-justify-center cide-ele-btn-content tw-w-full\">\r\n <!-- Left icon -->\r\n @if (leftIcon) {\r\n <span class=\"tw-icon-container tw-mr-2\">\r\n <i class=\"tw-text-base material-symbols-outlined\">{{leftIcon}}</i>\r\n </span>\r\n }\r\n \r\n <!-- Loading spinner -->\r\n @if (loading) {\r\n <cide-ele-spinner\r\n class=\"tw-inline-block tw-my-1 tw-mr-2\" \r\n size=\"xs\"\r\n [ngClass]=\"{'cide-pill-disabled': (disabled || loading)}\">\r\n </cide-ele-spinner>\r\n }\r\n \r\n <!-- Button label or content -->\r\n @if (label) {\r\n <span>{{ label }}</span>\r\n } @else {\r\n <span class=\"tw-flex tw-items-center tw-justify-center\">\r\n <ng-content></ng-content>\r\n </span>\r\n }\r\n \r\n <!-- Right icon -->\r\n @if (rightIcon) {\r\n <span class=\"tw-icon-container tw-ml-2\">\r\n <i class=\"tw-text-base material-symbols-outlined\">{{rightIcon}}</i>\r\n </span>\r\n }\r\n \r\n <!-- Spacer for spinner when loading to maintain button width -->\r\n @if (loading) {\r\n <span class=\"tw-w-6 tw-p-1 tw-mr-2\"></span>\r\n }\r\n</div>\r\n", styles: [":host{position:relative;background-color:var(--cide-theme-color-brand-primary, #3b82f6);cursor:pointer;font-family:inherit;font-weight:500;outline:none;overflow:hidden;transition:all .2s ease-in-out;border:none;text-align:center;vertical-align:middle;text-decoration:none;line-height:1.5;letter-spacing:.025em;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;-webkit-appearance:none;-moz-appearance:none;appearance:none}:host:focus-visible{outline:2px solid var(--cide-theme-color-brand-primary);outline-offset:2px;box-shadow:0 0 0 3px #3b82f64d}:host:hover:not(:disabled){background-color:var(--cide-theme-color-brand-primary-hover, #2563eb);transform:translateY(-1px)}:host:active:not(:disabled){transform:translateY(1px)}:host.cide-button-disabled{background-color:var(--cide-button-background-disabled, #9ca3af)!important;cursor:var(--cide-button-cursor-disabled, not-allowed)!important;opacity:.7;pointer-events:none;transform:none!important}:host.xs{font-size:.75rem;padding:.2rem .4rem}:host.sm{font-size:.875rem;padding:.25rem .5rem}:host.md{font-size:1rem;padding:.25rem .75rem}:host.lg{font-size:1.125rem;padding:.75rem 1rem}:host.xl{font-size:1.25rem;padding:1rem 1.5rem}:host.rounded{border-radius:.5rem}:host.pill{border-radius:9999px}:host.square{border-radius:0}:host.elevation-none{box-shadow:none}:host.elevation-low{box-shadow:0 1px 2px #0000000d}:host.elevation-medium{box-shadow:0 4px 6px -1px #0000001a,0 2px 4px -1px #0000000f}:host.elevation-high{box-shadow:0 10px 15px -3px #0000001a,0 4px 6px -2px #0000000d}.ripple-effect{position:absolute;border-radius:50%;background-color:#fff6;transform:scale(0);animation:ripple .6s linear;pointer-events:none}@keyframes ripple{to{transform:scale(4);opacity:0}}:host{color:#fff}:host.secondary{background-color:var(--cide-theme-secondary-color, #4b5563);color:#fff}:host.secondary:hover:not(:disabled){background-color:var(--cide-theme-secondary-color-hover, #374151)}:host.success{background-color:var(--cide-theme-success-color, #10b981);color:#fff}:host.success:hover:not(:disabled){background-color:var(--cide-theme-success-color-hover, #059669)}:host.danger{background-color:var(--cide-theme-danger-color, #ef4444);color:#fff}:host.danger:hover:not(:disabled){background-color:var(--cide-theme-danger-color-hover, #dc2626)}:host.warning{background-color:var(--cide-theme-warning-color, #f59e0b);color:#fff}:host.warning:hover:not(:disabled){background-color:var(--cide-theme-warning-color-hover, #d97706)}:host.info{background-color:var(--cide-theme-info-color, #3b82f6);color:#fff}:host.info:hover:not(:disabled){background-color:var(--cide-theme-info-color-hover, #2563eb)}:host.outline{background-color:transparent;color:var(--cide-theme-color-brand-primary, #3b82f6);border:1px solid var(--cide-theme-color-brand-primary, #3b82f6)}:host.outline:hover:not(:disabled){background-color:#3b82f60d}:host.outline.secondary{color:var(--cide-theme-secondary-color, #4b5563);border-color:var(--cide-theme-secondary-color, #4b5563)}:host.outline.success{color:var(--cide-theme-success-color, #10b981);border-color:var(--cide-theme-success-color, #10b981)}:host.outline.danger{color:var(--cide-theme-danger-color, #ef4444);border-color:var(--cide-theme-danger-color, #ef4444)}:host.outline.warning{color:var(--cide-theme-warning-color, #f59e0b);border-color:var(--cide-theme-warning-color, #f59e0b)}:host.outline.info{color:var(--cide-theme-info-color, #3b82f6);border-color:var(--cide-theme-info-color, #3b82f6)}:host.text{background-color:transparent;color:var(--cide-theme-color-brand-primary, #3b82f6);border:none;padding-left:.5rem;padding-right:.5rem}:host.text:hover:not(:disabled){background-color:#3b82f60d;text-decoration:underline}:host.text.secondary{color:var(--cide-theme-secondary-color, #4b5563)}:host.text.success{color:var(--cide-theme-success-color, #10b981)}:host.text.danger{color:var(--cide-theme-danger-color, #ef4444)}:host.text.warning{color:var(--cide-theme-warning-color, #f59e0b)}:host.text.info{color:var(--cide-theme-info-color, #3b82f6)}:host.link{background-color:transparent;color:var(--cide-theme-color-brand-primary, #3b82f6);border:none;text-decoration:underline;padding:0;font-weight:400}:host.link:hover:not(:disabled){color:var(--cide-theme-color-brand-primary-hover, #2563eb);text-decoration:underline}:host.link.secondary{color:var(--cide-theme-secondary-color, #4b5563)}:host.link.success{color:var(--cide-theme-success-color, #10b981)}:host.link.danger{color:var(--cide-theme-danger-color, #ef4444)}:host.link.warning{color:var(--cide-theme-warning-color, #f59e0b)}:host.link.info{color:var(--cide-theme-info-color, #3b82f6)}.tw-icon-container{display:inline-flex;align-items:center;justify-content:center}\n"] }]
201
+ }], ctorParameters: () => [{ type: i1.CideElementsService }, { type: i2.Router }, { type: i0.Renderer2 }, { type: i0.ElementRef }], propDecorators: { label: [{
202
+ type: Input
203
+ }], variant: [{
204
+ type: Input
205
+ }], size: [{
206
+ type: Input
207
+ }], type: [{
208
+ type: Input
209
+ }], shape: [{
210
+ type: Input
211
+ }], elevation: [{
212
+ type: Input
213
+ }], disabled: [{
214
+ type: Input
215
+ }], id: [{
216
+ type: Input
217
+ }], loading: [{
218
+ type: Input
219
+ }], fullWidth: [{
220
+ type: Input
221
+ }], leftIcon: [{
222
+ type: Input
223
+ }], rightIcon: [{
224
+ type: Input
225
+ }], customClass: [{
226
+ type: Input
227
+ }], tooltip: [{
228
+ type: Input
229
+ }], ariaLabel: [{
230
+ type: Input
231
+ }], testId: [{
232
+ type: Input
233
+ }], routerLink: [{
234
+ type: Input
235
+ }], routerExtras: [{
236
+ type: Input
237
+ }], preventDoubleClick: [{
238
+ type: Input
239
+ }], animated: [{
240
+ type: Input
241
+ }], btnClick: [{
242
+ type: Output
243
+ }], doubleClick: [{
244
+ type: Output
245
+ }], onClick: [{
246
+ type: HostListener,
247
+ args: ['click', ['$event']]
248
+ }] } });
249
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cide-ele-button.component.js","sourceRoot":"","sources":["../../../../../../projects/cloud-ide-element/src/lib/elements/button/cide-ele-button.component.ts","../../../../../../projects/cloud-ide-element/src/lib/elements/button/cide-ele-button.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, ElementRef, EventEmitter, HostListener, Input, OnChanges, OnDestroy, OnInit, Output, Renderer2, SimpleChanges } from '@angular/core';\r\nimport { CommonModule } from '@angular/common';\r\nimport { CideElementsService } from '../../../public-api';\r\nimport { CideSpinnerComponent } from '../spinner/spinner.component';\r\nimport { NavigationExtras, Router } from '@angular/router';\r\nimport { Subject, debounceTime, takeUntil } from 'rxjs';\r\n\r\nexport type ButtonVariant = 'primary' | 'secondary' | 'outline' | 'text' | 'link' | 'success' | 'danger' | 'warning' | 'info' | 'ghost';\r\nexport type ButtonSize = 'xs' | 'sm' | 'md' | 'lg' | 'xl';\r\nexport type ButtonType = 'button' | 'submit' | 'reset';\r\nexport type ButtonShape = 'default' | 'rounded' | 'pill' | 'square';\r\nexport type ButtonElevation = 'none' | 'low' | 'medium' | 'high';\r\n\r\n/**\r\n * Advanced Angular button component that provides a comprehensive button implementation\r\n * with loading state, styling variants, animations, tooltips, and other features.\r\n * \r\n * This component uses proper Angular templating to avoid hydration issues (NG0500).\r\n */\r\n@Component({\r\n  // eslint-disable-next-line @angular-eslint/component-selector\r\n  selector: 'button[cideEleButton], a[cideEleButton]',\r\n  standalone: true,\r\n  imports: [CommonModule, CideSpinnerComponent],\r\n  templateUrl: './cide-ele-button.component.html',\r\n  styleUrls: ['./cide-ele-button.component.scss'],\r\n  host: {\r\n    'class': 'cide-button tw-rounded-md tw-text-white tw-py-0.5 tw-select-none tw-justify-around tw-flex',\r\n    '[class.cide-button-disabled]': 'disabled || loading',\r\n    '[attr.disabled]': '(disabled || loading) ? true : null',\r\n    '[class.primary]': 'variant === \"primary\"',\r\n    '[class.secondary]': 'variant === \"secondary\"',\r\n    '[class.outline]': 'variant === \"outline\"',\r\n    '[class.text]': 'variant === \"text\"',\r\n    '[class.link]': 'variant === \"link\"',\r\n    '[class.success]': 'variant === \"success\"',\r\n    '[class.danger]': 'variant === \"danger\"',\r\n    '[class.warning]': 'variant === \"warning\"',\r\n    '[class.info]': 'variant === \"info\"',\r\n    '[class.xs]': 'size === \"xs\"',\r\n    '[class.sm]': 'size === \"sm\"',\r\n    '[class.md]': 'size === \"md\"',\r\n    '[class.lg]': 'size === \"lg\"',\r\n    '[class.xl]': 'size === \"xl\"',\r\n    '[class.rounded]': 'shape === \"rounded\"',\r\n    '[class.pill]': 'shape === \"pill\"',\r\n    '[class.square]': 'shape === \"square\"',\r\n    '[class.elevation-none]': 'elevation === \"none\"',\r\n    '[class.elevation-low]': 'elevation === \"low\"',\r\n    '[class.elevation-medium]': 'elevation === \"medium\"',\r\n    '[class.elevation-high]': 'elevation === \"high\"',\r\n    '[attr.type]': 'type',\r\n    '[attr.aria-label]': 'ariaLabel',\r\n    '[attr.aria-disabled]': 'disabled || loading',\r\n    '[attr.data-testid]': 'testId',\r\n    '[attr.title]': 'tooltip',\r\n    '[class]': 'customClass'\r\n  }\r\n})\r\n\r\nexport class CideEleButtonComponent implements OnInit, OnChanges, OnDestroy {\r\n  private destroy$ = new Subject<void>();\r\n  private clickCount = 0;\r\n  private clickDebouncer$ = new Subject<MouseEvent>();\r\n\r\n  constructor(\r\n    private elementService: CideElementsService,\r\n    private router: Router,\r\n    private renderer: Renderer2,\r\n    private elementRef: ElementRef\r\n  ) {\r\n    // Setup debounced click handler to prevent double-clicks\r\n    this.clickDebouncer$\r\n      .pipe(\r\n        debounceTime(300),\r\n        takeUntil(this.destroy$)\r\n      )\r\n      .subscribe({\r\n        next: (event) => {\r\n          this.handleClick(event);\r\n        }\r\n      });\r\n  }\r\n  \r\n  // INPUTS\r\n  /** \r\n   * @description Label of button like Submit, Update, Add, etc.  \r\n   * Note: it is only used in case you did not set label between tags (via ng-content)\r\n   */\r\n  @Input() label: string = \"\";\r\n  \r\n  /** @description Button style variant */\r\n  @Input() variant: ButtonVariant = 'primary';\r\n  \r\n  /** @description Button size */\r\n  @Input() size: ButtonSize = 'md';\r\n  \r\n  /** @description Button HTML type attribute */\r\n  @Input() type: ButtonType = 'button';\r\n  \r\n  /** @description Button shape */\r\n  @Input() shape: ButtonShape = 'default';\r\n  \r\n  /** @description Button shadow elevation */\r\n  @Input() elevation: ButtonElevation = 'medium';\r\n  \r\n  /** @description Button disabled state */\r\n  @Input() disabled: boolean | null = false;\r\n  \r\n  /** @description Unique ID of control, used to differentiate elements */\r\n  @Input() id: string = '';\r\n  \r\n  /** @description Enables loading state with spinner */\r\n  @Input() loading: boolean = false;\r\n  \r\n  /** @description Makes button take full width of container */\r\n  @Input() fullWidth: boolean = false;\r\n  \r\n  /** @description Icon to display on the left side of the button */\r\n  @Input() leftIcon: string = '';\r\n  \r\n  /** @description Icon to display on the right side of the button */\r\n  @Input() rightIcon: string = '';\r\n  \r\n  /** @description Additional CSS classes to apply */\r\n  @Input() customClass: string = '';\r\n  \r\n  /** @description Tooltip text to display on hover */\r\n  @Input() tooltip: string = '';\r\n  \r\n  /** @description Aria label for accessibility */\r\n  @Input() ariaLabel: string = '';\r\n  \r\n  /** @description Test ID for automated testing */\r\n  @Input() testId: string = '';\r\n  \r\n  /** @description Route to navigate to when clicked (for link buttons) */\r\n  @Input() routerLink: string | unknown[] = '';\r\n  \r\n  /** @description Router navigation extras */\r\n  @Input() routerExtras: NavigationExtras = {};\r\n  \r\n  /** @description Prevent multiple rapid clicks (debounce) */\r\n  @Input() preventDoubleClick: boolean = true;\r\n  \r\n  /** @description Animation on click */\r\n  @Input() animated: boolean = true;\r\n  \r\n  // OUTPUTS\r\n  /**\r\n   * @description Click event of button\r\n   */\r\n  @Output() btnClick: EventEmitter<MouseEvent> = new EventEmitter();\r\n  \r\n  /**\r\n   * @description Double click event\r\n   */\r\n  @Output() doubleClick: EventEmitter<MouseEvent> = new EventEmitter();\r\n\r\n  @HostListener('click', ['$event'])\r\n  onClick(event: MouseEvent): void {\r\n    if (this.disabled || this.loading) {\r\n      event.preventDefault();\r\n      event.stopPropagation();\r\n      return;\r\n    }\r\n    \r\n    if (this.animated) {\r\n      this.addClickAnimation();\r\n    }\r\n    \r\n    if (this.preventDoubleClick) {\r\n      this.clickDebouncer$.next(event);\r\n    } else {\r\n      this.handleClick(event);\r\n    }\r\n  }\r\n  \r\n  private handleClick(event: MouseEvent): void {\r\n    // Handle navigation for routerLink\r\n    if (this.routerLink && this.routerLink.length) {\r\n      event.preventDefault();\r\n      this.router.navigate(\r\n        Array.isArray(this.routerLink) ? this.routerLink : [this.routerLink],\r\n        this.routerExtras\r\n      );\r\n    }\r\n    \r\n    // Handle double-click detection\r\n    this.clickCount++;\r\n    setTimeout(() => {\r\n      if (this.clickCount === 2) {\r\n        this.doubleClick.emit(event);\r\n      }\r\n      this.clickCount = 0;\r\n    }, 300);\r\n    \r\n    // Emit standard click event\r\n    this.btnClick.emit(event);\r\n  }\r\n  \r\n  private addClickAnimation(): void {\r\n    // Add ripple effect\r\n    const ripple = this.renderer.createElement('span');\r\n    this.renderer.addClass(ripple, 'ripple-effect');\r\n    \r\n    // Position the ripple at click position\r\n    const rect = this.elementRef.nativeElement.getBoundingClientRect();\r\n    const size = Math.max(rect.width, rect.height);\r\n    \r\n    // Set ripple style\r\n    this.renderer.setStyle(ripple, 'width', `${size}px`);\r\n    this.renderer.setStyle(ripple, 'height', `${size}px`);\r\n    this.renderer.setStyle(ripple, 'left', `${0}px`);\r\n    this.renderer.setStyle(ripple, 'top', `${0}px`);\r\n    \r\n    // Add ripple to button\r\n    this.renderer.appendChild(this.elementRef.nativeElement, ripple);\r\n    \r\n    // Modern ES2022+ pattern: Use Promise-based delay\r\n    const delay = (ms: number) => new Promise(resolve => setTimeout(resolve, ms));\r\n    delay(600).then(() => {\r\n      this.renderer.removeChild(this.elementRef.nativeElement, ripple);\r\n    });\r\n  }\r\n\r\n  ngOnInit(): void {\r\n    if (this.id) {\r\n      this.getControlData();\r\n    }\r\n    \r\n    // Set aria-label if not explicitly provided\r\n    if (!this.ariaLabel && this.label) {\r\n      this.ariaLabel = this.label;\r\n    }\r\n  }\r\n  \r\n  ngOnChanges(changes: SimpleChanges): void {\r\n    // Update aria-label when label changes\r\n    if (changes['label'] && !this.ariaLabel) {\r\n      this.ariaLabel = changes['label'].currentValue;\r\n    }\r\n  }\r\n  \r\n  ngOnDestroy(): void {\r\n    this.destroy$.next();\r\n    this.destroy$.complete();\r\n  }\r\n\r\n  async getControlData() {\r\n    const cide_element_data = await this.elementService?.getElementData({ sype_key: this.id });\r\n    if (cide_element_data) {\r\n      this.label = cide_element_data?.sype_label as string;\r\n    }\r\n  }\r\n}\r\n","<!-- Button content container -->\r\n<div class=\"tw-flex tw-items-center tw-justify-center cide-ele-btn-content tw-w-full\">\r\n  <!-- Left icon -->\r\n  @if (leftIcon) {\r\n    <span class=\"tw-icon-container tw-mr-2\">\r\n      <i class=\"tw-text-base material-symbols-outlined\">{{leftIcon}}</i>\r\n    </span>\r\n  }\r\n  \r\n  <!-- Loading spinner -->\r\n  @if (loading) {\r\n    <cide-ele-spinner\r\n      class=\"tw-inline-block tw-my-1 tw-mr-2\" \r\n      size=\"xs\"\r\n      [ngClass]=\"{'cide-pill-disabled': (disabled || loading)}\">\r\n    </cide-ele-spinner>\r\n  }\r\n  \r\n  <!-- Button label or content -->\r\n  @if (label) {\r\n    <span>{{ label }}</span>\r\n  } @else {\r\n    <span class=\"tw-flex tw-items-center tw-justify-center\">\r\n      <ng-content></ng-content>\r\n    </span>\r\n  }\r\n  \r\n  <!-- Right icon -->\r\n  @if (rightIcon) {\r\n    <span class=\"tw-icon-container tw-ml-2\">\r\n      <i class=\"tw-text-base material-symbols-outlined\">{{rightIcon}}</i>\r\n    </span>\r\n  }\r\n  \r\n  <!-- Spacer for spinner when loading to maintain button width -->\r\n  @if (loading) {\r\n    <span class=\"tw-w-6 tw-p-1 tw-mr-2\"></span>\r\n  }\r\n</div>\r\n"]}
@@ -0,0 +1,83 @@
1
+ import { CommonModule } from '@angular/common';
2
+ import { Component, EventEmitter, Input, Output, forwardRef } from '@angular/core';
3
+ import { NG_VALUE_ACCESSOR, FormsModule } from '@angular/forms';
4
+ import * as i0 from "@angular/core";
5
+ import * as i1 from "@angular/common";
6
+ export class CideEleFileInputComponent {
7
+ constructor() {
8
+ this.label = 'Choose file';
9
+ this.accept = '';
10
+ this.multiple = false;
11
+ this.disabled = false;
12
+ this.helperText = '';
13
+ this.errorText = '';
14
+ this.id = Math.random().toString(36).substring(2, 10);
15
+ this.fileChange = new EventEmitter();
16
+ // eslint-disable-next-line @typescript-eslint/no-unused-vars
17
+ this.files = null;
18
+ this.fileNames = [];
19
+ // eslint-disable-next-line @typescript-eslint/no-unused-vars
20
+ this.onChange = (files) => { };
21
+ this.onTouched = () => { };
22
+ }
23
+ writeValue(files) {
24
+ this.files = files;
25
+ this.fileNames = files ? Array.from(files).map(f => f.name) : [];
26
+ }
27
+ registerOnChange(fn) {
28
+ this.onChange = fn;
29
+ }
30
+ registerOnTouched(fn) {
31
+ this.onTouched = fn;
32
+ }
33
+ setDisabledState(isDisabled) {
34
+ this.disabled = isDisabled;
35
+ }
36
+ onFileSelected(event) {
37
+ const input = event.target;
38
+ this.files = input.files;
39
+ this.fileNames = this.files ? Array.from(this.files).map(f => f.name) : [];
40
+ this.onChange(this.files);
41
+ this.fileChange.emit(this.files);
42
+ this.onTouched();
43
+ }
44
+ clearFiles() {
45
+ this.files = null;
46
+ this.fileNames = [];
47
+ this.onChange(null);
48
+ this.fileChange.emit(null);
49
+ }
50
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: CideEleFileInputComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
51
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.7", type: CideEleFileInputComponent, isStandalone: true, selector: "cide-ele-file-input", inputs: { label: "label", accept: "accept", multiple: "multiple", disabled: "disabled", helperText: "helperText", errorText: "errorText" }, outputs: { fileChange: "fileChange" }, providers: [
52
+ {
53
+ provide: NG_VALUE_ACCESSOR,
54
+ useExisting: forwardRef(() => CideEleFileInputComponent),
55
+ multi: true
56
+ }
57
+ ], ngImport: i0, template: "<div class=\"cide-file-input\">\r\n <label *ngIf=\"label\" class=\"cide-file-input-label\" [attr.for]=\"'cide-file-input-' + id\">{{ label }}</label>\r\n <div class=\"cide-file-input-wrapper\">\r\n <input\r\n type=\"file\"\r\n [attr.id]=\"'cide-file-input-' + id\"\r\n [attr.accept]=\"accept\"\r\n [attr.multiple]=\"multiple ? true : null\"\r\n [disabled]=\"disabled\"\r\n (change)=\"onFileSelected($event)\"\r\n class=\"cide-file-input-element\"\r\n />\r\n <button *ngIf=\"fileNames.length\" type=\"button\" class=\"cide-file-input-clear\" (click)=\"clearFiles()\">\r\n Clear\r\n </button>\r\n </div>\r\n <div *ngIf=\"fileNames.length\" class=\"cide-file-input-files\">\r\n <span *ngFor=\"let name of fileNames\">{{ name }}</span>\r\n </div>\r\n <div *ngIf=\"errorText\" class=\"cide-file-input-error\">{{ errorText }}</div>\r\n <div *ngIf=\"helperText && !errorText\" class=\"cide-file-input-helper\">{{ helperText }}</div>\r\n</div> ", styles: [".cide-file-input{display:flex;flex-direction:column;gap:.5rem}.cide-file-input-label{font-weight:500;margin-bottom:.25rem}.cide-file-input-wrapper{display:flex;align-items:center;gap:.5rem}.cide-file-input-element{flex:1}.cide-file-input-clear{background:none;border:none;color:#d32f2f;cursor:pointer;font-size:.9rem}.cide-file-input-files{font-size:.95rem;color:#333;margin-top:.25rem}.cide-file-input-error{color:#d32f2f;font-size:.9rem}.cide-file-input-helper{color:#666;font-size:.9rem}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "ngmodule", type: FormsModule }] }); }
58
+ }
59
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: CideEleFileInputComponent, decorators: [{
60
+ type: Component,
61
+ args: [{ selector: 'cide-ele-file-input', standalone: true, imports: [CommonModule, FormsModule], providers: [
62
+ {
63
+ provide: NG_VALUE_ACCESSOR,
64
+ useExisting: forwardRef(() => CideEleFileInputComponent),
65
+ multi: true
66
+ }
67
+ ], template: "<div class=\"cide-file-input\">\r\n <label *ngIf=\"label\" class=\"cide-file-input-label\" [attr.for]=\"'cide-file-input-' + id\">{{ label }}</label>\r\n <div class=\"cide-file-input-wrapper\">\r\n <input\r\n type=\"file\"\r\n [attr.id]=\"'cide-file-input-' + id\"\r\n [attr.accept]=\"accept\"\r\n [attr.multiple]=\"multiple ? true : null\"\r\n [disabled]=\"disabled\"\r\n (change)=\"onFileSelected($event)\"\r\n class=\"cide-file-input-element\"\r\n />\r\n <button *ngIf=\"fileNames.length\" type=\"button\" class=\"cide-file-input-clear\" (click)=\"clearFiles()\">\r\n Clear\r\n </button>\r\n </div>\r\n <div *ngIf=\"fileNames.length\" class=\"cide-file-input-files\">\r\n <span *ngFor=\"let name of fileNames\">{{ name }}</span>\r\n </div>\r\n <div *ngIf=\"errorText\" class=\"cide-file-input-error\">{{ errorText }}</div>\r\n <div *ngIf=\"helperText && !errorText\" class=\"cide-file-input-helper\">{{ helperText }}</div>\r\n</div> ", styles: [".cide-file-input{display:flex;flex-direction:column;gap:.5rem}.cide-file-input-label{font-weight:500;margin-bottom:.25rem}.cide-file-input-wrapper{display:flex;align-items:center;gap:.5rem}.cide-file-input-element{flex:1}.cide-file-input-clear{background:none;border:none;color:#d32f2f;cursor:pointer;font-size:.9rem}.cide-file-input-files{font-size:.95rem;color:#333;margin-top:.25rem}.cide-file-input-error{color:#d32f2f;font-size:.9rem}.cide-file-input-helper{color:#666;font-size:.9rem}\n"] }]
68
+ }], propDecorators: { label: [{
69
+ type: Input
70
+ }], accept: [{
71
+ type: Input
72
+ }], multiple: [{
73
+ type: Input
74
+ }], disabled: [{
75
+ type: Input
76
+ }], helperText: [{
77
+ type: Input
78
+ }], errorText: [{
79
+ type: Input
80
+ }], fileChange: [{
81
+ type: Output
82
+ }] } });
83
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZmlsZS1pbnB1dC5jb21wb25lbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9jbG91ZC1pZGUtZWxlbWVudC9zcmMvbGliL2VsZW1lbnRzL2ZpbGUtaW5wdXQvZmlsZS1pbnB1dC5jb21wb25lbnQudHMiLCIuLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9jbG91ZC1pZGUtZWxlbWVudC9zcmMvbGliL2VsZW1lbnRzL2ZpbGUtaW5wdXQvZmlsZS1pbnB1dC5jb21wb25lbnQuaHRtbCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0saUJBQWlCLENBQUM7QUFDL0MsT0FBTyxFQUFFLFNBQVMsRUFBRSxZQUFZLEVBQUUsS0FBSyxFQUFFLE1BQU0sRUFBRSxVQUFVLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDbkYsT0FBTyxFQUF3QixpQkFBaUIsRUFBRSxXQUFXLEVBQUUsTUFBTSxnQkFBZ0IsQ0FBQzs7O0FBZ0J0RixNQUFNLE9BQU8seUJBQXlCO0lBZHRDO1FBZVcsVUFBSyxHQUFXLGFBQWEsQ0FBQztRQUM5QixXQUFNLEdBQVcsRUFBRSxDQUFDO1FBQ3BCLGFBQVEsR0FBWSxLQUFLLENBQUM7UUFDMUIsYUFBUSxHQUFZLEtBQUssQ0FBQztRQUMxQixlQUFVLEdBQVcsRUFBRSxDQUFDO1FBQ3hCLGNBQVMsR0FBVyxFQUFFLENBQUM7UUFFekIsT0FBRSxHQUFXLElBQUksQ0FBQyxNQUFNLEVBQUUsQ0FBQyxRQUFRLENBQUMsRUFBRSxDQUFDLENBQUMsU0FBUyxDQUFDLENBQUMsRUFBRSxFQUFFLENBQUMsQ0FBQztRQUV0RCxlQUFVLEdBQUcsSUFBSSxZQUFZLEVBQW1CLENBQUM7UUFFM0QsNkRBQTZEO1FBQzdELFVBQUssR0FBb0IsSUFBSSxDQUFDO1FBQzlCLGNBQVMsR0FBYSxFQUFFLENBQUM7UUFFekIsNkRBQTZEO1FBQ3JELGFBQVEsR0FBRyxDQUFDLEtBQXNCLEVBQUUsRUFBRSxHQUFFLENBQUMsQ0FBQztRQUMxQyxjQUFTLEdBQUcsR0FBRyxFQUFFLEdBQUUsQ0FBQyxDQUFDO0tBa0M5QjtJQWhDQyxVQUFVLENBQUMsS0FBc0I7UUFDL0IsSUFBSSxDQUFDLEtBQUssR0FBRyxLQUFLLENBQUM7UUFDbkIsSUFBSSxDQUFDLFNBQVMsR0FBRyxLQUFLLENBQUMsQ0FBQyxDQUFDLEtBQUssQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLENBQUMsR0FBRyxDQUFDLENBQUMsQ0FBQyxFQUFFLENBQUMsQ0FBQyxDQUFDLElBQUksQ0FBQyxDQUFDLENBQUMsQ0FBQyxFQUFFLENBQUM7SUFDbkUsQ0FBQztJQUVELGdCQUFnQixDQUFDLEVBQW9DO1FBQ25ELElBQUksQ0FBQyxRQUFRLEdBQUcsRUFBRSxDQUFDO0lBQ3JCLENBQUM7SUFFRCxpQkFBaUIsQ0FBQyxFQUFjO1FBQzlCLElBQUksQ0FBQyxTQUFTLEdBQUcsRUFBRSxDQUFDO0lBQ3RCLENBQUM7SUFFRCxnQkFBZ0IsQ0FBQyxVQUFtQjtRQUNsQyxJQUFJLENBQUMsUUFBUSxHQUFHLFVBQVUsQ0FBQztJQUM3QixDQUFDO0lBRUQsY0FBYyxDQUFDLEtBQVk7UUFDekIsTUFBTSxLQUFLLEdBQUcsS0FBSyxDQUFDLE1BQTBCLENBQUM7UUFDL0MsSUFBSSxDQUFDLEtBQUssR0FBRyxLQUFLLENBQUMsS0FBSyxDQUFDO1FBQ3pCLElBQUksQ0FBQyxTQUFTLEdBQUcsSUFBSSxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsS0FBSyxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLENBQUMsR0FBRyxDQUFDLENBQUMsQ0FBQyxFQUFFLENBQUMsQ0FBQyxDQUFDLElBQUksQ0FBQyxDQUFDLENBQUMsQ0FBQyxFQUFFLENBQUM7UUFDM0UsSUFBSSxDQUFDLFFBQVEsQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLENBQUM7UUFDMUIsSUFBSSxDQUFDLFVBQVUsQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxDQUFDO1FBQ2pDLElBQUksQ0FBQyxTQUFTLEVBQUUsQ0FBQztJQUNuQixDQUFDO0lBRUQsVUFBVTtRQUNSLElBQUksQ0FBQyxLQUFLLEdBQUcsSUFBSSxDQUFDO1FBQ2xCLElBQUksQ0FBQyxTQUFTLEdBQUcsRUFBRSxDQUFDO1FBQ3BCLElBQUksQ0FBQyxRQUFRLENBQUMsSUFBSSxDQUFDLENBQUM7UUFDcEIsSUFBSSxDQUFDLFVBQVUsQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLENBQUM7SUFDN0IsQ0FBQzs4R0FuRFUseUJBQXlCO2tHQUF6Qix5QkFBeUIscVBBUnpCO1lBQ1Q7Z0JBQ0UsT0FBTyxFQUFFLGlCQUFpQjtnQkFDMUIsV0FBVyxFQUFFLFVBQVUsQ0FBQyxHQUFHLEVBQUUsQ0FBQyx5QkFBeUIsQ0FBQztnQkFDeEQsS0FBSyxFQUFFLElBQUk7YUFDWjtTQUNGLDBCQ2hCSCx3K0JBcUJPLHFpQkRaSyxZQUFZLCtQQUFFLFdBQVc7OzJGQVN4Qix5QkFBeUI7a0JBZHJDLFNBQVM7K0JBQ0UscUJBQXFCLGNBQ25CLElBQUksV0FHUCxDQUFDLFlBQVksRUFBRSxXQUFXLENBQUMsYUFDekI7d0JBQ1Q7NEJBQ0UsT0FBTyxFQUFFLGlCQUFpQjs0QkFDMUIsV0FBVyxFQUFFLFVBQVUsQ0FBQyxHQUFHLEVBQUUsMEJBQTBCLENBQUM7NEJBQ3hELEtBQUssRUFBRSxJQUFJO3lCQUNaO3FCQUNGOzhCQUdRLEtBQUs7c0JBQWIsS0FBSztnQkFDRyxNQUFNO3NCQUFkLEtBQUs7Z0JBQ0csUUFBUTtzQkFBaEIsS0FBSztnQkFDRyxRQUFRO3NCQUFoQixLQUFLO2dCQUNHLFVBQVU7c0JBQWxCLEtBQUs7Z0JBQ0csU0FBUztzQkFBakIsS0FBSztnQkFJSSxVQUFVO3NCQUFuQixNQUFNIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ29tbW9uTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvY29tbW9uJztcclxuaW1wb3J0IHsgQ29tcG9uZW50LCBFdmVudEVtaXR0ZXIsIElucHV0LCBPdXRwdXQsIGZvcndhcmRSZWYgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcclxuaW1wb3J0IHsgQ29udHJvbFZhbHVlQWNjZXNzb3IsIE5HX1ZBTFVFX0FDQ0VTU09SLCBGb3Jtc01vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL2Zvcm1zJztcclxuXHJcbkBDb21wb25lbnQoe1xyXG4gIHNlbGVjdG9yOiAnY2lkZS1lbGUtZmlsZS1pbnB1dCcsXHJcbiAgc3RhbmRhbG9uZTogdHJ1ZSxcclxuICB0ZW1wbGF0ZVVybDogJy4vZmlsZS1pbnB1dC5jb21wb25lbnQuaHRtbCcsXHJcbiAgc3R5bGVVcmw6ICcuL2ZpbGUtaW5wdXQuY29tcG9uZW50LnNjc3MnLFxyXG4gIGltcG9ydHM6IFtDb21tb25Nb2R1bGUsIEZvcm1zTW9kdWxlXSxcclxuICBwcm92aWRlcnM6IFtcclxuICAgIHtcclxuICAgICAgcHJvdmlkZTogTkdfVkFMVUVfQUNDRVNTT1IsXHJcbiAgICAgIHVzZUV4aXN0aW5nOiBmb3J3YXJkUmVmKCgpID0+IENpZGVFbGVGaWxlSW5wdXRDb21wb25lbnQpLFxyXG4gICAgICBtdWx0aTogdHJ1ZVxyXG4gICAgfVxyXG4gIF1cclxufSlcclxuZXhwb3J0IGNsYXNzIENpZGVFbGVGaWxlSW5wdXRDb21wb25lbnQgaW1wbGVtZW50cyBDb250cm9sVmFsdWVBY2Nlc3NvciB7XHJcbiAgQElucHV0KCkgbGFiZWw6IHN0cmluZyA9ICdDaG9vc2UgZmlsZSc7XHJcbiAgQElucHV0KCkgYWNjZXB0OiBzdHJpbmcgPSAnJztcclxuICBASW5wdXQoKSBtdWx0aXBsZTogYm9vbGVhbiA9IGZhbHNlO1xyXG4gIEBJbnB1dCgpIGRpc2FibGVkOiBib29sZWFuID0gZmFsc2U7XHJcbiAgQElucHV0KCkgaGVscGVyVGV4dDogc3RyaW5nID0gJyc7XHJcbiAgQElucHV0KCkgZXJyb3JUZXh0OiBzdHJpbmcgPSAnJztcclxuXHJcbiAgcHVibGljIGlkOiBzdHJpbmcgPSBNYXRoLnJhbmRvbSgpLnRvU3RyaW5nKDM2KS5zdWJzdHJpbmcoMiwgMTApO1xyXG5cclxuICBAT3V0cHV0KCkgZmlsZUNoYW5nZSA9IG5ldyBFdmVudEVtaXR0ZXI8RmlsZUxpc3QgfCBudWxsPigpO1xyXG5cclxuICAvLyBlc2xpbnQtZGlzYWJsZS1uZXh0LWxpbmUgQHR5cGVzY3JpcHQtZXNsaW50L25vLXVudXNlZC12YXJzXHJcbiAgZmlsZXM6IEZpbGVMaXN0IHwgbnVsbCA9IG51bGw7XHJcbiAgZmlsZU5hbWVzOiBzdHJpbmdbXSA9IFtdO1xyXG5cclxuICAvLyBlc2xpbnQtZGlzYWJsZS1uZXh0LWxpbmUgQHR5cGVzY3JpcHQtZXNsaW50L25vLXVudXNlZC12YXJzXHJcbiAgcHJpdmF0ZSBvbkNoYW5nZSA9IChmaWxlczogRmlsZUxpc3QgfCBudWxsKSA9PiB7fTtcclxuICBwcml2YXRlIG9uVG91Y2hlZCA9ICgpID0+IHt9O1xyXG5cclxuICB3cml0ZVZhbHVlKGZpbGVzOiBGaWxlTGlzdCB8IG51bGwpOiB2b2lkIHtcclxuICAgIHRoaXMuZmlsZXMgPSBmaWxlcztcclxuICAgIHRoaXMuZmlsZU5hbWVzID0gZmlsZXMgPyBBcnJheS5mcm9tKGZpbGVzKS5tYXAoZiA9PiBmLm5hbWUpIDogW107XHJcbiAgfVxyXG5cclxuICByZWdpc3Rlck9uQ2hhbmdlKGZuOiAoZmlsZXM6IEZpbGVMaXN0IHwgbnVsbCkgPT4gdm9pZCk6IHZvaWQge1xyXG4gICAgdGhpcy5vbkNoYW5nZSA9IGZuO1xyXG4gIH1cclxuXHJcbiAgcmVnaXN0ZXJPblRvdWNoZWQoZm46ICgpID0+IHZvaWQpOiB2b2lkIHtcclxuICAgIHRoaXMub25Ub3VjaGVkID0gZm47XHJcbiAgfVxyXG5cclxuICBzZXREaXNhYmxlZFN0YXRlKGlzRGlzYWJsZWQ6IGJvb2xlYW4pOiB2b2lkIHtcclxuICAgIHRoaXMuZGlzYWJsZWQgPSBpc0Rpc2FibGVkO1xyXG4gIH1cclxuXHJcbiAgb25GaWxlU2VsZWN0ZWQoZXZlbnQ6IEV2ZW50KSB7XHJcbiAgICBjb25zdCBpbnB1dCA9IGV2ZW50LnRhcmdldCBhcyBIVE1MSW5wdXRFbGVtZW50O1xyXG4gICAgdGhpcy5maWxlcyA9IGlucHV0LmZpbGVzO1xyXG4gICAgdGhpcy5maWxlTmFtZXMgPSB0aGlzLmZpbGVzID8gQXJyYXkuZnJvbSh0aGlzLmZpbGVzKS5tYXAoZiA9PiBmLm5hbWUpIDogW107XHJcbiAgICB0aGlzLm9uQ2hhbmdlKHRoaXMuZmlsZXMpO1xyXG4gICAgdGhpcy5maWxlQ2hhbmdlLmVtaXQodGhpcy5maWxlcyk7XHJcbiAgICB0aGlzLm9uVG91Y2hlZCgpO1xyXG4gIH1cclxuXHJcbiAgY2xlYXJGaWxlcygpIHtcclxuICAgIHRoaXMuZmlsZXMgPSBudWxsO1xyXG4gICAgdGhpcy5maWxlTmFtZXMgPSBbXTtcclxuICAgIHRoaXMub25DaGFuZ2UobnVsbCk7XHJcbiAgICB0aGlzLmZpbGVDaGFuZ2UuZW1pdChudWxsKTtcclxuICB9XHJcbn0gIiwiPGRpdiBjbGFzcz1cImNpZGUtZmlsZS1pbnB1dFwiPlxyXG4gIDxsYWJlbCAqbmdJZj1cImxhYmVsXCIgY2xhc3M9XCJjaWRlLWZpbGUtaW5wdXQtbGFiZWxcIiBbYXR0ci5mb3JdPVwiJ2NpZGUtZmlsZS1pbnB1dC0nICsgaWRcIj57eyBsYWJlbCB9fTwvbGFiZWw+XHJcbiAgPGRpdiBjbGFzcz1cImNpZGUtZmlsZS1pbnB1dC13cmFwcGVyXCI+XHJcbiAgICA8aW5wdXRcclxuICAgICAgdHlwZT1cImZpbGVcIlxyXG4gICAgICBbYXR0ci5pZF09XCInY2lkZS1maWxlLWlucHV0LScgKyBpZFwiXHJcbiAgICAgIFthdHRyLmFjY2VwdF09XCJhY2NlcHRcIlxyXG4gICAgICBbYXR0ci5tdWx0aXBsZV09XCJtdWx0aXBsZSA/IHRydWUgOiBudWxsXCJcclxuICAgICAgW2Rpc2FibGVkXT1cImRpc2FibGVkXCJcclxuICAgICAgKGNoYW5nZSk9XCJvbkZpbGVTZWxlY3RlZCgkZXZlbnQpXCJcclxuICAgICAgY2xhc3M9XCJjaWRlLWZpbGUtaW5wdXQtZWxlbWVudFwiXHJcbiAgICAvPlxyXG4gICAgPGJ1dHRvbiAqbmdJZj1cImZpbGVOYW1lcy5sZW5ndGhcIiB0eXBlPVwiYnV0dG9uXCIgY2xhc3M9XCJjaWRlLWZpbGUtaW5wdXQtY2xlYXJcIiAoY2xpY2spPVwiY2xlYXJGaWxlcygpXCI+XHJcbiAgICAgIENsZWFyXHJcbiAgICA8L2J1dHRvbj5cclxuICA8L2Rpdj5cclxuICA8ZGl2ICpuZ0lmPVwiZmlsZU5hbWVzLmxlbmd0aFwiIGNsYXNzPVwiY2lkZS1maWxlLWlucHV0LWZpbGVzXCI+XHJcbiAgICA8c3BhbiAqbmdGb3I9XCJsZXQgbmFtZSBvZiBmaWxlTmFtZXNcIj57eyBuYW1lIH19PC9zcGFuPlxyXG4gIDwvZGl2PlxyXG4gIDxkaXYgKm5nSWY9XCJlcnJvclRleHRcIiBjbGFzcz1cImNpZGUtZmlsZS1pbnB1dC1lcnJvclwiPnt7IGVycm9yVGV4dCB9fTwvZGl2PlxyXG4gIDxkaXYgKm5nSWY9XCJoZWxwZXJUZXh0ICYmICFlcnJvclRleHRcIiBjbGFzcz1cImNpZGUtZmlsZS1pbnB1dC1oZWxwZXJcIj57eyBoZWxwZXJUZXh0IH19PC9kaXY+XHJcbjwvZGl2PiAiXX0=