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
package/README.md CHANGED
@@ -1,24 +1,271 @@
1
- # CloudIdeElement
2
-
3
- This library was generated with [Angular CLI](https://github.com/angular/angular-cli) version 18.0.0.
4
-
5
- ## Code scaffolding
6
-
7
- Run `ng generate component component-name --project cloud-ide-element` to generate a new component. You can also use `ng generate directive|pipe|service|class|guard|interface|enum|module --project cloud-ide-element`.
8
- > Note: Don't forget to add `--project cloud-ide-element` or else it will be added to the default project in your `angular.json` file.
9
-
10
- ## Build
11
-
12
- Run `ng build cloud-ide-element` to build the project. The build artifacts will be stored in the `dist/` directory.
13
-
14
- ## Publishing
15
-
16
- After building your library with `ng build cloud-ide-element`, go to the dist folder `cd dist/cloud-ide-element` and run `npm publish`.
17
-
18
- ## Running unit tests
19
-
20
- Run `ng test cloud-ide-element` to execute the unit tests via [Karma](https://karma-runner.github.io).
21
-
22
- ## Further help
23
-
24
- To get more help on the Angular CLI use `ng help` or go check out the [Angular CLI Overview and Command Reference](https://angular.dev/tools/cli) page.
1
+ # cloud-ide-element
2
+
3
+ [![npm version](https://badge.fury.io/js/cloud-ide-element.svg)](https://badge.fury.io/js/cloud-ide-element)
4
+ [![License: MIT](https://img.shields.io/badge/License-MIT-yellow.svg)](https://opensource.org/licenses/MIT)
5
+ [![Angular](https://img.shields.io/badge/Angular-18+-red.svg)](https://angular.io/)
6
+ [![TypeScript](https://img.shields.io/badge/TypeScript-5.0+-blue.svg)](https://www.typescriptlang.org/)
7
+
8
+ A comprehensive Angular UI component library built for modern web applications. Featuring responsive design, TypeScript support, and Angular 18+ compatibility with signals and standalone components.
9
+
10
+ ## ✨ Features
11
+
12
+ - 🎨 **Modern Design** - Clean, responsive components with Tailwind CSS
13
+ - 🔧 **TypeScript** - Full type safety and IntelliSense support
14
+ - 📱 **Responsive** - Mobile-first design approach
15
+ - ⚡ **Angular 18+** - Built with latest Angular features (signals, control flow)
16
+ - 🧩 **Standalone** - Tree-shakable standalone components
17
+ - 🎯 **Accessible** - WCAG compliant components
18
+ - 🔥 **Performance** - Optimized for production use
19
+
20
+ ## 📦 Installation
21
+
22
+ ```bash
23
+ npm install cloud-ide-element
24
+ ```
25
+
26
+ ## 🚀 Quick Start
27
+
28
+ ### Import Components
29
+
30
+ ```typescript
31
+ import {
32
+ CideEleButtonComponent,
33
+ CideEleInputComponent,
34
+ CideEleTabComponent,
35
+ CideEleDataGridComponent
36
+ } from 'cloud-ide-element';
37
+
38
+ @Component({
39
+ standalone: true,
40
+ imports: [
41
+ CideEleButtonComponent,
42
+ CideEleInputComponent,
43
+ CideEleTabComponent,
44
+ CideEleDataGridComponent
45
+ ],
46
+ template: `
47
+ <cide-ele-button variant="primary" size="md">
48
+ Click me!
49
+ </cide-ele-button>
50
+
51
+ <cide-ele-input
52
+ label="Email"
53
+ type="email"
54
+ placeholder="Enter your email">
55
+ </cide-ele-input>
56
+
57
+ <cide-ele-tab [tabs]="tabs" (tabChange)="onTabChange($event)"></cide-ele-tab>
58
+
59
+ <cide-ele-data-grid [config]="gridConfig"></cide-ele-data-grid>
60
+ `
61
+ })
62
+ export class MyComponent { }
63
+ ```
64
+
65
+ ### Data Grid Example
66
+
67
+ ```typescript
68
+ import { Component, signal } from '@angular/core';
69
+ import { CideEleDataGridComponent, GridConfiguration } from 'cloud-ide-element';
70
+
71
+ @Component({
72
+ standalone: true,
73
+ imports: [CideEleDataGridComponent],
74
+ template: `<cide-ele-data-grid [config]="gridConfig()"></cide-ele-data-grid>`
75
+ })
76
+ export class UserGridComponent {
77
+ gridConfig = signal<GridConfiguration>({
78
+ id: 'users',
79
+ title: 'User Management',
80
+ columns: [
81
+ { key: 'name', header: 'Name', type: 'text', searchable: true },
82
+ { key: 'email', header: 'Email', type: 'text', searchable: true },
83
+ { key: 'isActive', header: 'Status', type: 'boolean' },
84
+ {
85
+ key: 'actions',
86
+ header: 'Actions',
87
+ type: 'actions',
88
+ actions: [
89
+ { key: 'edit', label: 'Edit', variant: 'primary', onClick: 'editUser' }
90
+ ]
91
+ }
92
+ ],
93
+ data: [
94
+ { id: 1, name: 'John Doe', email: 'john@example.com', isActive: true },
95
+ { id: 2, name: 'Jane Smith', email: 'jane@example.com', isActive: false }
96
+ ],
97
+ pagination: { enabled: true, pageSize: 10 },
98
+ search: { enabled: true, placeholder: 'Search users...' }
99
+ });
100
+ }
101
+ ```
102
+
103
+ ## 📚 Component Library
104
+
105
+ ### 🎯 Basic UI Elements
106
+
107
+ | Component | Description | Features |
108
+ |-----------|-------------|----------|
109
+ | **Button** | Customizable button component | Multiple variants, sizes, loading states |
110
+ | **Input** | Form input with validation | Text, email, password, number types |
111
+ | **Select** | Dropdown select component | Single/multi-select, searchable options |
112
+ | **Textarea** | Multi-line text input | Auto-resize, character counting |
113
+ | **Tab** | Navigation tab component | Simple, compact design with variants |
114
+ | **Icon** | SVG icon component | Customizable size and color |
115
+ | **Spinner** | Loading indicator | Multiple sizes and variants |
116
+
117
+ ### 🚀 Advanced Components
118
+
119
+ | Component | Description | Key Features |
120
+ |-----------|-------------|--------------|
121
+ | **Data Grid** | Enterprise-grade data table | Pagination, search, sorting, custom renderers, actions |
122
+
123
+ ### 🔧 Utilities
124
+
125
+ | Directive | Description | Use Case |
126
+ |-----------|-------------|----------|
127
+ | **Tooltip** | Hover tooltip directive | Help text and information display |
128
+ | **Resizer** | Element resize directive | Draggable resize functionality |
129
+
130
+ ## 📖 Documentation
131
+
132
+ ### Component Documentation
133
+
134
+ - [Button Component](./src/lib/elements/button/README.md) - Buttons, links, and action triggers
135
+ - [Input Component](./src/lib/elements/input/README.md) - Form inputs with validation
136
+ - [Select Component](./src/lib/elements/select/README.md) - Dropdown selects and multi-selects
137
+ - [Textarea Component](./src/lib/elements/textarea/README.md) - Multi-line text areas
138
+ - [Tab Component](./src/lib/elements/tab/README.md) - Navigation tabs for content organization
139
+ - [Icon Component](./src/lib/elements/icon/README.md) - SVG icons and graphics
140
+ - [Spinner Component](./src/lib/elements/spinner/README.md) - Loading states and progress
141
+
142
+ ### Advanced Components
143
+
144
+ - [**Data Grid Component**](./src/lib/components/data-grid/README.md) - Complete data table solution with advanced features
145
+
146
+ ### Directives
147
+
148
+ - [Tooltip Directive](./src/lib/utils/directives/tooltip/README.md) - Hover tooltips and help text
149
+ - [Resizer Directive](./src/lib/utils/directives/resizer/README.md) - Draggable element resizing
150
+
151
+ ## 🎨 Styling & Theming
152
+
153
+ This library uses Tailwind CSS for styling. Import the theme variables:
154
+
155
+ ```scss
156
+ @import 'cloud-ide-element/style';
157
+ ```
158
+
159
+ ### Custom Theme Variables
160
+
161
+ ```scss
162
+ // Override default theme variables
163
+ :root {
164
+ --cide-primary-color: #3b82f6;
165
+ --cide-secondary-color: #6b7280;
166
+ --cide-border-radius: 6px;
167
+ }
168
+ ```
169
+
170
+ ## 🔧 Configuration
171
+
172
+ ### Angular.json Setup
173
+
174
+ Add to your `angular.json` styles array:
175
+
176
+ ```json
177
+ {
178
+ "styles": [
179
+ "node_modules/cloud-ide-element/lib/assets/css/cide-ele-style.scss"
180
+ ]
181
+ }
182
+ ```
183
+
184
+ ### Module Setup (Optional)
185
+
186
+ For non-standalone usage:
187
+
188
+ ```typescript
189
+ import { NgModule } from '@angular/core';
190
+ import {
191
+ CideEleButtonComponent,
192
+ CideEleInputComponent,
193
+ CideEleDataGridComponent
194
+ } from 'cloud-ide-element';
195
+
196
+ @NgModule({
197
+ imports: [
198
+ CideEleButtonComponent,
199
+ CideEleInputComponent,
200
+ CideEleDataGridComponent
201
+ ],
202
+ exports: [
203
+ CideEleButtonComponent,
204
+ CideEleInputComponent,
205
+ CideEleDataGridComponent
206
+ ]
207
+ })
208
+ export class CloudIdeElementModule { }
209
+ ```
210
+
211
+ ## 🌟 Key Highlights
212
+
213
+ ### Data Grid Features
214
+ - **📊 Rich Data Display** - Tables with sorting, filtering, and pagination
215
+ - **🔍 Advanced Search** - Multi-column search with debouncing
216
+ - **⚡ Performance** - Virtual scrolling for large datasets
217
+ - **🎨 Custom Rendering** - Custom cell renderers and formatters
218
+ - **📱 Responsive** - Mobile-optimized layouts
219
+ - **🔧 Action Buttons** - Configurable row actions with conditions
220
+
221
+ ### Form Components
222
+ - **✅ Validation** - Built-in form validation support
223
+ - **♿ Accessibility** - WCAG 2.1 AA compliant
224
+ - **🎯 Consistent** - Unified design language across all components
225
+ - **🔧 Flexible** - Extensive customization options
226
+
227
+ ## 🤝 Contributing
228
+
229
+ We welcome contributions! Please see our [Contributing Guide](CONTRIBUTING.md) for details.
230
+
231
+ ### Development Setup
232
+
233
+ ```bash
234
+ # Clone the repository
235
+ git clone https://github.com/your-org/cloud-ide-element.git
236
+
237
+ # Install dependencies
238
+ npm install
239
+
240
+ # Start development server
241
+ ng serve
242
+
243
+ # Build the library
244
+ ng build cloud-ide-element
245
+
246
+ # Run tests
247
+ ng test cloud-ide-element
248
+ ```
249
+
250
+ ## 📄 License
251
+
252
+ This project is licensed under the MIT License - see the [LICENSE](LICENSE) file for details.
253
+
254
+ ## 🚀 Roadmap
255
+
256
+ - [ ] Tree Table Component
257
+ - [ ] Calendar/Date Picker
258
+ - [ ] File Upload Component
259
+ - [ ] Chart Components
260
+ - [ ] Rich Text Editor
261
+ - [ ] Dashboard Widgets
262
+
263
+ ## 💬 Support
264
+
265
+ - 📧 Email: support@cide-lms.com
266
+ - 🐛 Issues: [GitHub Issues](https://github.com/your-org/cloud-ide-element/issues)
267
+ - 📖 Documentation: [Full Documentation](https://docs.cide-lms.com)
268
+
269
+ ---
270
+
271
+ Made with ❤️ by the CIDE-LMS Team
@@ -0,0 +1,182 @@
1
+ import { Component, signal, computed, inject } from '@angular/core';
2
+ import { CommonModule } from '@angular/common';
3
+ import { CideIconComponent } from '../../elements/icon/icon.component';
4
+ import { ConfirmationService } from '../../services/confirmation.service';
5
+ import * as i0 from "@angular/core";
6
+ import * as i1 from "@angular/common";
7
+ export class CideEleConfirmationModalComponent {
8
+ constructor() {
9
+ this.confirmationService = inject(ConfirmationService);
10
+ // Signals
11
+ this.hasActiveConfirmation = this.confirmationService.hasActiveConfirmation;
12
+ this.currentRequest = this.confirmationService.getCurrentRequest();
13
+ this.customData = signal(null);
14
+ // Computed
15
+ this.getHeaderClass = computed(() => {
16
+ const request = this.currentRequest();
17
+ return request ? `header-${request.type}` : '';
18
+ });
19
+ this.getIconClass = computed(() => {
20
+ const request = this.currentRequest();
21
+ return request ? request.type : 'info';
22
+ });
23
+ this.getConfirmButtonClass = computed(() => {
24
+ const request = this.currentRequest();
25
+ if (!request)
26
+ return 'btn-primary';
27
+ const classMap = {
28
+ danger: 'btn-danger',
29
+ warning: 'btn-warning',
30
+ info: 'btn-primary',
31
+ success: 'btn-success'
32
+ };
33
+ return classMap[request.type] || 'btn-primary';
34
+ });
35
+ }
36
+ onBackdropClick(event) {
37
+ // Only close if clicking the backdrop itself, not its children
38
+ if (event.target === event.currentTarget) {
39
+ this.onCancel();
40
+ }
41
+ }
42
+ onCancel() {
43
+ this.confirmationService.cancelCurrentRequest();
44
+ }
45
+ onConfirm() {
46
+ // If there's custom data, pass it to the confirmation
47
+ const data = this.customData();
48
+ this.confirmationService.confirmCurrentRequest(data);
49
+ this.customData.set(null); // Reset custom data
50
+ }
51
+ // Method to update custom data (called by custom templates)
52
+ updateCustomData(data) {
53
+ this.customData.set(data);
54
+ }
55
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: CideEleConfirmationModalComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
56
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.7", type: CideEleConfirmationModalComponent, isStandalone: true, selector: "cide-ele-confirmation-modal", ngImport: i0, template: `
57
+ <!-- Backdrop -->
58
+ <div
59
+ *ngIf="hasActiveConfirmation()"
60
+ class="confirmation-backdrop"
61
+ (click)="onBackdropClick($event)"
62
+ [@backdropAnimation]>
63
+ </div>
64
+
65
+ <!-- Modal -->
66
+ <div
67
+ *ngIf="hasActiveConfirmation()"
68
+ class="confirmation-modal"
69
+ [@modalAnimation]>
70
+
71
+ <!-- Modal Header -->
72
+ <div class="modal-header" [class]="getHeaderClass()">
73
+ <div class="header-content">
74
+ <div class="icon-container" [class]="getIconClass()">
75
+ <cide-ele-icon
76
+ size="lg">
77
+ {{ currentRequest()?.icon || 'info' }}
78
+ </cide-ele-icon>
79
+ </div>
80
+ <h2 class="modal-title">{{ currentRequest()?.title }}</h2>
81
+ </div>
82
+ </div>
83
+
84
+ <!-- Modal Body -->
85
+ <div class="modal-body">
86
+ <p class="modal-message">{{ currentRequest()?.message }}</p>
87
+
88
+ <!-- Custom Template Slot -->
89
+ <div *ngIf="currentRequest()?.customTemplate" class="custom-content">
90
+ <ng-container
91
+ *ngTemplateOutlet="currentRequest()?.customTemplate!; context: { $implicit: customData }">
92
+ </ng-container>
93
+ </div>
94
+ </div>
95
+
96
+ <!-- Modal Footer -->
97
+ <div class="modal-footer">
98
+ <button
99
+ type="button"
100
+ class="btn btn-secondary"
101
+ (click)="onCancel()">
102
+ {{ currentRequest()?.cancelText || 'Cancel' }}
103
+ </button>
104
+
105
+ <button
106
+ type="button"
107
+ class="btn"
108
+ [class]="getConfirmButtonClass()"
109
+ (click)="onConfirm()">
110
+ {{ currentRequest()?.confirmText || 'Confirm' }}
111
+ </button>
112
+ </div>
113
+ </div>
114
+ `, isInline: true, styles: [".confirmation-backdrop{position:fixed;inset:0;background-color:#00000080;-webkit-backdrop-filter:blur(4px);backdrop-filter:blur(4px);z-index:10000;display:flex;align-items:center;justify-content:center}.confirmation-modal{background:#fff;border-radius:12px;box-shadow:0 20px 25px -5px #0000001a,0 10px 10px -5px #0000000a;max-width:480px;width:90%;max-height:90vh;overflow:hidden;z-index:10001;position:relative}.modal-header{padding:24px 24px 16px;border-bottom:1px solid #e5e7eb}.header-content{display:flex;align-items:center;gap:16px}.icon-container{width:48px;height:48px;border-radius:50%;display:flex;align-items:center;justify-content:center;flex-shrink:0}.icon-container.danger{background-color:#fef2f2;color:#dc2626}.icon-container.warning{background-color:#fffbeb;color:#d97706}.icon-container.info{background-color:#eff6ff;color:#2563eb}.icon-container.success{background-color:#f0fdf4;color:#16a34a}.modal-title{margin:0;font-size:20px;font-weight:600;color:#111827;line-height:1.4}.modal-body{padding:16px 24px 24px}.modal-message{margin:0 0 16px;font-size:16px;line-height:1.5;color:#6b7280}.custom-content{margin-top:16px;padding-top:16px;border-top:1px solid #f3f4f6}.modal-footer{padding:16px 24px 24px;display:flex;gap:12px;justify-content:flex-end;border-top:1px solid #e5e7eb}.btn{padding:10px 20px;border-radius:8px;font-size:14px;font-weight:500;border:none;cursor:pointer;transition:all .2s ease;min-width:80px}.btn:hover{transform:translateY(-1px)}.btn:active{transform:translateY(0)}.btn-secondary{background-color:#f9fafb;color:#374151;border:1px solid #d1d5db}.btn-secondary:hover{background-color:#f3f4f6;border-color:#9ca3af}.btn-primary{background-color:#2563eb;color:#fff}.btn-primary:hover{background-color:#1d4ed8}.btn-danger{background-color:#dc2626;color:#fff}.btn-danger:hover{background-color:#b91c1c}.btn-warning{background-color:#d97706;color:#fff}.btn-warning:hover{background-color:#b45309}.btn-success{background-color:#16a34a;color:#fff}.btn-success:hover{background-color:#15803d}@media (max-width: 640px){.confirmation-modal{width:95%;margin:20px}.modal-header{padding:20px 20px 12px}.modal-body{padding:12px 20px 20px}.modal-footer{padding:12px 20px 20px;flex-direction:column-reverse}.btn{width:100%;justify-content:center}.header-content{flex-direction:column;text-align:center;gap:12px}.modal-title{font-size:18px}}.btn:focus{outline:2px solid #2563eb;outline-offset:2px}.modal-enter{opacity:0;transform:scale(.95) translateY(-10px)}.modal-enter-active{opacity:1;transform:scale(1) translateY(0);transition:all .2s ease-out}.modal-exit{opacity:1;transform:scale(1) translateY(0)}.modal-exit-active{opacity:0;transform:scale(.95) translateY(-10px);transition:all .15s ease-in}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "component", type: CideIconComponent, selector: "cide-ele-icon", inputs: ["size", "type", "toolTip"] }], animations: [
115
+ // Add animations here if needed
116
+ ] }); }
117
+ }
118
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.7", ngImport: i0, type: CideEleConfirmationModalComponent, decorators: [{
119
+ type: Component,
120
+ args: [{ selector: 'cide-ele-confirmation-modal', standalone: true, imports: [CommonModule, CideIconComponent], template: `
121
+ <!-- Backdrop -->
122
+ <div
123
+ *ngIf="hasActiveConfirmation()"
124
+ class="confirmation-backdrop"
125
+ (click)="onBackdropClick($event)"
126
+ [@backdropAnimation]>
127
+ </div>
128
+
129
+ <!-- Modal -->
130
+ <div
131
+ *ngIf="hasActiveConfirmation()"
132
+ class="confirmation-modal"
133
+ [@modalAnimation]>
134
+
135
+ <!-- Modal Header -->
136
+ <div class="modal-header" [class]="getHeaderClass()">
137
+ <div class="header-content">
138
+ <div class="icon-container" [class]="getIconClass()">
139
+ <cide-ele-icon
140
+ size="lg">
141
+ {{ currentRequest()?.icon || 'info' }}
142
+ </cide-ele-icon>
143
+ </div>
144
+ <h2 class="modal-title">{{ currentRequest()?.title }}</h2>
145
+ </div>
146
+ </div>
147
+
148
+ <!-- Modal Body -->
149
+ <div class="modal-body">
150
+ <p class="modal-message">{{ currentRequest()?.message }}</p>
151
+
152
+ <!-- Custom Template Slot -->
153
+ <div *ngIf="currentRequest()?.customTemplate" class="custom-content">
154
+ <ng-container
155
+ *ngTemplateOutlet="currentRequest()?.customTemplate!; context: { $implicit: customData }">
156
+ </ng-container>
157
+ </div>
158
+ </div>
159
+
160
+ <!-- Modal Footer -->
161
+ <div class="modal-footer">
162
+ <button
163
+ type="button"
164
+ class="btn btn-secondary"
165
+ (click)="onCancel()">
166
+ {{ currentRequest()?.cancelText || 'Cancel' }}
167
+ </button>
168
+
169
+ <button
170
+ type="button"
171
+ class="btn"
172
+ [class]="getConfirmButtonClass()"
173
+ (click)="onConfirm()">
174
+ {{ currentRequest()?.confirmText || 'Confirm' }}
175
+ </button>
176
+ </div>
177
+ </div>
178
+ `, animations: [
179
+ // Add animations here if needed
180
+ ], styles: [".confirmation-backdrop{position:fixed;inset:0;background-color:#00000080;-webkit-backdrop-filter:blur(4px);backdrop-filter:blur(4px);z-index:10000;display:flex;align-items:center;justify-content:center}.confirmation-modal{background:#fff;border-radius:12px;box-shadow:0 20px 25px -5px #0000001a,0 10px 10px -5px #0000000a;max-width:480px;width:90%;max-height:90vh;overflow:hidden;z-index:10001;position:relative}.modal-header{padding:24px 24px 16px;border-bottom:1px solid #e5e7eb}.header-content{display:flex;align-items:center;gap:16px}.icon-container{width:48px;height:48px;border-radius:50%;display:flex;align-items:center;justify-content:center;flex-shrink:0}.icon-container.danger{background-color:#fef2f2;color:#dc2626}.icon-container.warning{background-color:#fffbeb;color:#d97706}.icon-container.info{background-color:#eff6ff;color:#2563eb}.icon-container.success{background-color:#f0fdf4;color:#16a34a}.modal-title{margin:0;font-size:20px;font-weight:600;color:#111827;line-height:1.4}.modal-body{padding:16px 24px 24px}.modal-message{margin:0 0 16px;font-size:16px;line-height:1.5;color:#6b7280}.custom-content{margin-top:16px;padding-top:16px;border-top:1px solid #f3f4f6}.modal-footer{padding:16px 24px 24px;display:flex;gap:12px;justify-content:flex-end;border-top:1px solid #e5e7eb}.btn{padding:10px 20px;border-radius:8px;font-size:14px;font-weight:500;border:none;cursor:pointer;transition:all .2s ease;min-width:80px}.btn:hover{transform:translateY(-1px)}.btn:active{transform:translateY(0)}.btn-secondary{background-color:#f9fafb;color:#374151;border:1px solid #d1d5db}.btn-secondary:hover{background-color:#f3f4f6;border-color:#9ca3af}.btn-primary{background-color:#2563eb;color:#fff}.btn-primary:hover{background-color:#1d4ed8}.btn-danger{background-color:#dc2626;color:#fff}.btn-danger:hover{background-color:#b91c1c}.btn-warning{background-color:#d97706;color:#fff}.btn-warning:hover{background-color:#b45309}.btn-success{background-color:#16a34a;color:#fff}.btn-success:hover{background-color:#15803d}@media (max-width: 640px){.confirmation-modal{width:95%;margin:20px}.modal-header{padding:20px 20px 12px}.modal-body{padding:12px 20px 20px}.modal-footer{padding:12px 20px 20px;flex-direction:column-reverse}.btn{width:100%;justify-content:center}.header-content{flex-direction:column;text-align:center;gap:12px}.modal-title{font-size:18px}}.btn:focus{outline:2px solid #2563eb;outline-offset:2px}.modal-enter{opacity:0;transform:scale(.95) translateY(-10px)}.modal-enter-active{opacity:1;transform:scale(1) translateY(0);transition:all .2s ease-out}.modal-exit{opacity:1;transform:scale(1) translateY(0)}.modal-exit-active{opacity:0;transform:scale(.95) translateY(-10px);transition:all .15s ease-in}\n"] }]
181
+ }] });
182
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"confirmation-modal.component.js","sourceRoot":"","sources":["../../../../../../projects/cloud-ide-element/src/lib/components/confirmation-modal/confirmation-modal.component.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,MAAM,EAAE,QAAQ,EAAE,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,EAAE,MAAM,qCAAqC,CAAC;;;AA6S1E,MAAM,OAAO,iCAAiC;IA3S9C;QA4SU,wBAAmB,GAAG,MAAM,CAAC,mBAAmB,CAAC,CAAC;QAE1D,UAAU;QACV,0BAAqB,GAAG,IAAI,CAAC,mBAAmB,CAAC,qBAAqB,CAAC;QACvE,mBAAc,GAAG,IAAI,CAAC,mBAAmB,CAAC,iBAAiB,EAAE,CAAC;QAC9D,eAAU,GAAG,MAAM,CAAU,IAAI,CAAC,CAAC;QAEnC,WAAW;QACX,mBAAc,GAAG,QAAQ,CAAC,GAAG,EAAE;YAC7B,MAAM,OAAO,GAAG,IAAI,CAAC,cAAc,EAAE,CAAC;YACtC,OAAO,OAAO,CAAC,CAAC,CAAC,UAAU,OAAO,CAAC,IAAI,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC;QACjD,CAAC,CAAC,CAAC;QAEH,iBAAY,GAAG,QAAQ,CAAC,GAAG,EAAE;YAC3B,MAAM,OAAO,GAAG,IAAI,CAAC,cAAc,EAAE,CAAC;YACtC,OAAO,OAAO,CAAC,CAAC,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,MAAM,CAAC;QACzC,CAAC,CAAC,CAAC;QAEH,0BAAqB,GAAG,QAAQ,CAAC,GAAG,EAAE;YACpC,MAAM,OAAO,GAAG,IAAI,CAAC,cAAc,EAAE,CAAC;YACtC,IAAI,CAAC,OAAO;gBAAE,OAAO,aAAa,CAAC;YAEnC,MAAM,QAAQ,GAAG;gBACf,MAAM,EAAE,YAAY;gBACpB,OAAO,EAAE,aAAa;gBACtB,IAAI,EAAE,aAAa;gBACnB,OAAO,EAAE,aAAa;aACvB,CAAC;YAEF,OAAO,QAAQ,CAAC,OAAO,CAAC,IAAI,CAAC,IAAI,aAAa,CAAC;QACjD,CAAC,CAAC,CAAC;KAwBJ;IAtBC,eAAe,CAAC,KAAY;QAC1B,+DAA+D;QAC/D,IAAI,KAAK,CAAC,MAAM,KAAK,KAAK,CAAC,aAAa,EAAE,CAAC;YACzC,IAAI,CAAC,QAAQ,EAAE,CAAC;QAClB,CAAC;IACH,CAAC;IAED,QAAQ;QACN,IAAI,CAAC,mBAAmB,CAAC,oBAAoB,EAAE,CAAC;IAClD,CAAC;IAED,SAAS;QACP,sDAAsD;QACtD,MAAM,IAAI,GAAG,IAAI,CAAC,UAAU,EAAE,CAAC;QAC/B,IAAI,CAAC,mBAAmB,CAAC,qBAAqB,CAAC,IAAI,CAAC,CAAC;QACrD,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC,CAAC,oBAAoB;IACjD,CAAC;IAED,4DAA4D;IAC5D,gBAAgB,CAAC,IAAa;QAC5B,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;IAC5B,CAAC;8GAtDU,iCAAiC;kGAAjC,iCAAiC,uFAvSlC;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA0DT,gvFA3DS,YAAY,0SAAE,iBAAiB,iFAoS7B;QACV,gCAAgC;SACjC;;2FAEU,iCAAiC;kBA3S7C,SAAS;+BACE,6BAA6B,cAC3B,IAAI,WACP,CAAC,YAAY,EAAE,iBAAiB,CAAC,YAChC;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA0DT,cAyOW;oBACV,gCAAgC;qBACjC","sourcesContent":["import { Component, signal, computed, inject } from '@angular/core';\r\nimport { CommonModule } from '@angular/common';\r\nimport { CideIconComponent } from '../../elements/icon/icon.component';\r\nimport { ConfirmationService } from '../../services/confirmation.service';\r\n\r\n@Component({\r\n  selector: 'cide-ele-confirmation-modal',\r\n  standalone: true,\r\n  imports: [CommonModule, CideIconComponent],\r\n  template: `\r\n    <!-- Backdrop -->\r\n    <div \r\n      *ngIf=\"hasActiveConfirmation()\"\r\n      class=\"confirmation-backdrop\"\r\n      (click)=\"onBackdropClick($event)\"\r\n      [@backdropAnimation]>\r\n    </div>\r\n\r\n    <!-- Modal -->\r\n    <div \r\n      *ngIf=\"hasActiveConfirmation()\"\r\n      class=\"confirmation-modal\"\r\n      [@modalAnimation]>\r\n      \r\n      <!-- Modal Header -->\r\n      <div class=\"modal-header\" [class]=\"getHeaderClass()\">\r\n        <div class=\"header-content\">\r\n          <div class=\"icon-container\" [class]=\"getIconClass()\">\r\n            <cide-ele-icon \r\n              size=\"lg\">\r\n              {{ currentRequest()?.icon || 'info' }}\r\n            </cide-ele-icon>\r\n          </div>\r\n          <h2 class=\"modal-title\">{{ currentRequest()?.title }}</h2>\r\n        </div>\r\n      </div>\r\n\r\n      <!-- Modal Body -->\r\n      <div class=\"modal-body\">\r\n        <p class=\"modal-message\">{{ currentRequest()?.message }}</p>\r\n        \r\n        <!-- Custom Template Slot -->\r\n        <div *ngIf=\"currentRequest()?.customTemplate\" class=\"custom-content\">\r\n          <ng-container \r\n            *ngTemplateOutlet=\"currentRequest()?.customTemplate!; context: { $implicit: customData }\">\r\n          </ng-container>\r\n        </div>\r\n      </div>\r\n\r\n      <!-- Modal Footer -->\r\n      <div class=\"modal-footer\">\r\n        <button \r\n          type=\"button\"\r\n          class=\"btn btn-secondary\"\r\n          (click)=\"onCancel()\">\r\n          {{ currentRequest()?.cancelText || 'Cancel' }}\r\n        </button>\r\n        \r\n        <button \r\n          type=\"button\"\r\n          class=\"btn\"\r\n          [class]=\"getConfirmButtonClass()\"\r\n          (click)=\"onConfirm()\">\r\n          {{ currentRequest()?.confirmText || 'Confirm' }}\r\n        </button>\r\n      </div>\r\n    </div>\r\n  `,\r\n  styles: [`\r\n    .confirmation-backdrop {\r\n      position: fixed;\r\n      top: 0;\r\n      left: 0;\r\n      right: 0;\r\n      bottom: 0;\r\n      background-color: rgba(0, 0, 0, 0.5);\r\n      backdrop-filter: blur(4px);\r\n      z-index: 10000;\r\n      display: flex;\r\n      align-items: center;\r\n      justify-content: center;\r\n    }\r\n\r\n    .confirmation-modal {\r\n      background: white;\r\n      border-radius: 12px;\r\n      box-shadow: 0 20px 25px -5px rgba(0, 0, 0, 0.1), 0 10px 10px -5px rgba(0, 0, 0, 0.04);\r\n      max-width: 480px;\r\n      width: 90%;\r\n      max-height: 90vh;\r\n      overflow: hidden;\r\n      z-index: 10001;\r\n      position: relative;\r\n    }\r\n\r\n    .modal-header {\r\n      padding: 24px 24px 16px;\r\n      border-bottom: 1px solid #e5e7eb;\r\n    }\r\n\r\n    .header-content {\r\n      display: flex;\r\n      align-items: center;\r\n      gap: 16px;\r\n    }\r\n\r\n    .icon-container {\r\n      width: 48px;\r\n      height: 48px;\r\n      border-radius: 50%;\r\n      display: flex;\r\n      align-items: center;\r\n      justify-content: center;\r\n      flex-shrink: 0;\r\n    }\r\n\r\n    .icon-container.danger {\r\n      background-color: #fef2f2;\r\n      color: #dc2626;\r\n    }\r\n\r\n    .icon-container.warning {\r\n      background-color: #fffbeb;\r\n      color: #d97706;\r\n    }\r\n\r\n    .icon-container.info {\r\n      background-color: #eff6ff;\r\n      color: #2563eb;\r\n    }\r\n\r\n    .icon-container.success {\r\n      background-color: #f0fdf4;\r\n      color: #16a34a;\r\n    }\r\n\r\n    .modal-title {\r\n      margin: 0;\r\n      font-size: 20px;\r\n      font-weight: 600;\r\n      color: #111827;\r\n      line-height: 1.4;\r\n    }\r\n\r\n    .modal-body {\r\n      padding: 16px 24px 24px;\r\n    }\r\n\r\n    .modal-message {\r\n      margin: 0 0 16px 0;\r\n      font-size: 16px;\r\n      line-height: 1.5;\r\n      color: #6b7280;\r\n    }\r\n\r\n    .custom-content {\r\n      margin-top: 16px;\r\n      padding-top: 16px;\r\n      border-top: 1px solid #f3f4f6;\r\n    }\r\n\r\n    .modal-footer {\r\n      padding: 16px 24px 24px;\r\n      display: flex;\r\n      gap: 12px;\r\n      justify-content: flex-end;\r\n      border-top: 1px solid #e5e7eb;\r\n    }\r\n\r\n    .btn {\r\n      padding: 10px 20px;\r\n      border-radius: 8px;\r\n      font-size: 14px;\r\n      font-weight: 500;\r\n      border: none;\r\n      cursor: pointer;\r\n      transition: all 0.2s ease;\r\n      min-width: 80px;\r\n    }\r\n\r\n    .btn:hover {\r\n      transform: translateY(-1px);\r\n    }\r\n\r\n    .btn:active {\r\n      transform: translateY(0);\r\n    }\r\n\r\n    .btn-secondary {\r\n      background-color: #f9fafb;\r\n      color: #374151;\r\n      border: 1px solid #d1d5db;\r\n    }\r\n\r\n    .btn-secondary:hover {\r\n      background-color: #f3f4f6;\r\n      border-color: #9ca3af;\r\n    }\r\n\r\n    .btn-primary {\r\n      background-color: #2563eb;\r\n      color: white;\r\n    }\r\n\r\n    .btn-primary:hover {\r\n      background-color: #1d4ed8;\r\n    }\r\n\r\n    .btn-danger {\r\n      background-color: #dc2626;\r\n      color: white;\r\n    }\r\n\r\n    .btn-danger:hover {\r\n      background-color: #b91c1c;\r\n    }\r\n\r\n    .btn-warning {\r\n      background-color: #d97706;\r\n      color: white;\r\n    }\r\n\r\n    .btn-warning:hover {\r\n      background-color: #b45309;\r\n    }\r\n\r\n    .btn-success {\r\n      background-color: #16a34a;\r\n      color: white;\r\n    }\r\n\r\n    .btn-success:hover {\r\n      background-color: #15803d;\r\n    }\r\n\r\n    /* Responsive Design */\r\n    @media (max-width: 640px) {\r\n      .confirmation-modal {\r\n        width: 95%;\r\n        margin: 20px;\r\n      }\r\n\r\n      .modal-header {\r\n        padding: 20px 20px 12px;\r\n      }\r\n\r\n      .modal-body {\r\n        padding: 12px 20px 20px;\r\n      }\r\n\r\n      .modal-footer {\r\n        padding: 12px 20px 20px;\r\n        flex-direction: column-reverse;\r\n      }\r\n\r\n      .btn {\r\n        width: 100%;\r\n        justify-content: center;\r\n      }\r\n\r\n      .header-content {\r\n        flex-direction: column;\r\n        text-align: center;\r\n        gap: 12px;\r\n      }\r\n\r\n      .modal-title {\r\n        font-size: 18px;\r\n      }\r\n    }\r\n\r\n    /* Focus styles for accessibility */\r\n    .btn:focus {\r\n      outline: 2px solid #2563eb;\r\n      outline-offset: 2px;\r\n    }\r\n\r\n    /* Animation classes */\r\n    .modal-enter {\r\n      opacity: 0;\r\n      transform: scale(0.95) translateY(-10px);\r\n    }\r\n\r\n    .modal-enter-active {\r\n      opacity: 1;\r\n      transform: scale(1) translateY(0);\r\n      transition: all 0.2s ease-out;\r\n    }\r\n\r\n    .modal-exit {\r\n      opacity: 1;\r\n      transform: scale(1) translateY(0);\r\n    }\r\n\r\n    .modal-exit-active {\r\n      opacity: 0;\r\n      transform: scale(0.95) translateY(-10px);\r\n      transition: all 0.15s ease-in;\r\n    }\r\n  `],\r\n  animations: [\r\n    // Add animations here if needed\r\n  ]\r\n})\r\nexport class CideEleConfirmationModalComponent {\r\n  private confirmationService = inject(ConfirmationService);\r\n\r\n  // Signals\r\n  hasActiveConfirmation = this.confirmationService.hasActiveConfirmation;\r\n  currentRequest = this.confirmationService.getCurrentRequest();\r\n  customData = signal<unknown>(null);\r\n\r\n  // Computed\r\n  getHeaderClass = computed(() => {\r\n    const request = this.currentRequest();\r\n    return request ? `header-${request.type}` : '';\r\n  });\r\n\r\n  getIconClass = computed(() => {\r\n    const request = this.currentRequest();\r\n    return request ? request.type : 'info';\r\n  });\r\n\r\n  getConfirmButtonClass = computed(() => {\r\n    const request = this.currentRequest();\r\n    if (!request) return 'btn-primary';\r\n    \r\n    const classMap = {\r\n      danger: 'btn-danger',\r\n      warning: 'btn-warning',\r\n      info: 'btn-primary',\r\n      success: 'btn-success'\r\n    };\r\n    \r\n    return classMap[request.type] || 'btn-primary';\r\n  });\r\n\r\n  onBackdropClick(event: Event): void {\r\n    // Only close if clicking the backdrop itself, not its children\r\n    if (event.target === event.currentTarget) {\r\n      this.onCancel();\r\n    }\r\n  }\r\n\r\n  onCancel(): void {\r\n    this.confirmationService.cancelCurrentRequest();\r\n  }\r\n\r\n  onConfirm(): void {\r\n    // If there's custom data, pass it to the confirmation\r\n    const data = this.customData();\r\n    this.confirmationService.confirmCurrentRequest(data);\r\n    this.customData.set(null); // Reset custom data\r\n  }\r\n\r\n  // Method to update custom data (called by custom templates)\r\n  updateCustomData(data: unknown): void {\r\n    this.customData.set(data);\r\n  }\r\n}\r\n"]}