pdm-ui-kit 0.1.49 → 0.2.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.
- package/README.md +189 -2
- package/esm2020/lib/components/alert-dialog/alert-dialog.component.mjs +25 -7
- package/esm2020/lib/components/breadcrumb/breadcrumb.component.mjs +37 -4
- package/esm2020/lib/components/calendar/calendar.component.mjs +3 -3
- package/esm2020/lib/components/card/card.component.mjs +36 -53
- package/esm2020/lib/components/command/command.component.mjs +3 -3
- package/esm2020/lib/components/context-menu/context-menu.component.mjs +16 -8
- package/esm2020/lib/components/data-table/data-table.component.mjs +214 -16
- package/esm2020/lib/components/dialog/dialog.component.mjs +133 -17
- package/esm2020/lib/components/draggable-table/draggable-table.component.mjs +300 -0
- package/esm2020/lib/components/drawer/drawer.component.mjs +138 -10
- package/esm2020/lib/components/dropdown-menu/dropdown-menu.component.mjs +6 -3
- package/esm2020/lib/components/hover-card/hover-card.component.mjs +3 -3
- package/esm2020/lib/components/menubar/menubar.component.mjs +38 -7
- package/esm2020/lib/components/navigation-menu/navigation-menu.component.mjs +25 -3
- package/esm2020/lib/components/pagination/pagination.component.mjs +3 -3
- package/esm2020/lib/components/popover/popover.component.mjs +19 -11
- package/esm2020/lib/components/select/select.component.mjs +8 -4
- package/esm2020/lib/components/sheet/sheet.component.mjs +88 -11
- package/esm2020/lib/components/sidebar/sidebar.component.mjs +52 -5
- package/esm2020/lib/components/table/table.component.mjs +152 -188
- package/esm2020/lib/components/tabs/tabs.component.mjs +3 -3
- package/esm2020/lib/components/tooltip/tooltip.component.mjs +3 -3
- package/esm2020/lib/overlay/pdm-outside-click.directive.mjs +86 -0
- package/esm2020/lib/pdm-ui-kit.module.mjs +9 -1
- package/esm2020/lib/utils/responsive.mjs +143 -0
- package/esm2020/lib/utils/z-index.mjs +93 -0
- package/esm2020/public-api.mjs +5 -1
- package/fesm2015/pdm-ui-kit.mjs +1625 -370
- package/fesm2015/pdm-ui-kit.mjs.map +1 -1
- package/fesm2020/pdm-ui-kit.mjs +1620 -367
- package/fesm2020/pdm-ui-kit.mjs.map +1 -1
- package/lib/components/alert-dialog/alert-dialog.component.d.ts +9 -1
- package/lib/components/breadcrumb/breadcrumb.component.d.ts +23 -1
- package/lib/components/card/card.component.d.ts +32 -19
- package/lib/components/context-menu/context-menu.component.d.ts +6 -3
- package/lib/components/data-table/data-table.component.d.ts +172 -14
- package/lib/components/dialog/dialog.component.d.ts +35 -1
- package/lib/components/draggable-table/draggable-table.component.d.ts +74 -0
- package/lib/components/drawer/drawer.component.d.ts +67 -3
- package/lib/components/menubar/menubar.component.d.ts +10 -2
- package/lib/components/navigation-menu/navigation-menu.component.d.ts +22 -1
- package/lib/components/popover/popover.component.d.ts +6 -3
- package/lib/components/sheet/sheet.component.d.ts +34 -1
- package/lib/components/sidebar/sidebar.component.d.ts +39 -1
- package/lib/components/table/table.component.d.ts +46 -25
- package/lib/overlay/pdm-outside-click.directive.d.ts +40 -0
- package/lib/pdm-ui-kit.module.d.ts +42 -40
- package/lib/utils/responsive.d.ts +107 -0
- package/lib/utils/z-index.d.ts +73 -0
- package/package.json +5 -3
- package/public-api.d.ts +4 -0
|
@@ -1,11 +1,42 @@
|
|
|
1
1
|
import { ChangeDetectionStrategy, Component, EventEmitter, HostListener, Input, Output } from '@angular/core';
|
|
2
|
+
import { responsive } from '../../utils/responsive';
|
|
3
|
+
import { Z_INDEX } from '../../utils/z-index';
|
|
2
4
|
import * as i0 from "@angular/core";
|
|
3
5
|
import * as i1 from "@angular/common";
|
|
6
|
+
/**
|
|
7
|
+
* Modal/Dialog component con soporte responsive
|
|
8
|
+
*
|
|
9
|
+
* MEJORADO en v0.2.0:
|
|
10
|
+
* - Modo 'responsive' (default): fullscreen en mobile, modal en desktop
|
|
11
|
+
* - Tamaños predefinidos: sm, md, lg, xl
|
|
12
|
+
* - Mejor manejo de scroll en mobile
|
|
13
|
+
*
|
|
14
|
+
* @example
|
|
15
|
+
* // Responsive (recomendado)
|
|
16
|
+
* <pdm-dialog [open]="isOpen" size="responsive">
|
|
17
|
+
* <p>Content</p>
|
|
18
|
+
* </pdm-dialog>
|
|
19
|
+
*
|
|
20
|
+
* @example
|
|
21
|
+
* // Tamaño fijo
|
|
22
|
+
* <pdm-dialog [open]="isOpen" size="lg">
|
|
23
|
+
* <p>Content</p>
|
|
24
|
+
* </pdm-dialog>
|
|
25
|
+
*/
|
|
4
26
|
export class PdmDialogComponent {
|
|
5
27
|
constructor() {
|
|
6
28
|
this.open = false;
|
|
7
29
|
this.variant = 'default';
|
|
8
|
-
|
|
30
|
+
/**
|
|
31
|
+
* Tamaño del dialog
|
|
32
|
+
* - responsive: fullscreen mobile, modal desktop (recomendado)
|
|
33
|
+
* - sm: 400px max
|
|
34
|
+
* - md: 500px max
|
|
35
|
+
* - lg: 640px max (default)
|
|
36
|
+
* - xl: 800px max
|
|
37
|
+
* - desktop/mobile/mobile-fullscreen: legacy, deprecado
|
|
38
|
+
*/
|
|
39
|
+
this.size = 'responsive';
|
|
9
40
|
this.title = 'Edit profile';
|
|
10
41
|
this.description = '';
|
|
11
42
|
this.closeOnBackdrop = true;
|
|
@@ -44,47 +75,132 @@ export class PdmDialogComponent {
|
|
|
44
75
|
}
|
|
45
76
|
}
|
|
46
77
|
get panelClassName() {
|
|
78
|
+
// Legacy sizes (backward compatibility)
|
|
79
|
+
if (this.size === 'desktop') {
|
|
80
|
+
return this.buildClasses(['max-w-[640px]', 'max-h-[calc(100vh-2rem)]', 'rounded-[10px]']);
|
|
81
|
+
}
|
|
82
|
+
if (this.size === 'mobile') {
|
|
83
|
+
return this.buildClasses(['max-w-[320px]', 'min-h-[240px]', 'rounded-[10px]']);
|
|
84
|
+
}
|
|
85
|
+
if (this.size === 'mobile-fullscreen') {
|
|
86
|
+
return this.buildClasses(['max-w-[320px]', 'h-[min(100dvh,640px)]', 'rounded-none', 'sm:rounded-[10px]']);
|
|
87
|
+
}
|
|
88
|
+
// New responsive mode (recomendado)
|
|
89
|
+
if (this.size === 'responsive') {
|
|
90
|
+
return this.buildClasses([
|
|
91
|
+
// Mobile: fullscreen con bordes redondeados solo arriba
|
|
92
|
+
'w-full',
|
|
93
|
+
'h-full',
|
|
94
|
+
'max-h-[100dvh]',
|
|
95
|
+
'rounded-t-[10px]',
|
|
96
|
+
'sm:rounded-[10px]',
|
|
97
|
+
// Desktop: modal centrado
|
|
98
|
+
'sm:w-auto',
|
|
99
|
+
'sm:h-auto',
|
|
100
|
+
'sm:max-w-[640px]',
|
|
101
|
+
'sm:max-h-[calc(100vh-4rem)]'
|
|
102
|
+
]);
|
|
103
|
+
}
|
|
104
|
+
// New size options
|
|
105
|
+
const sizeMap = {
|
|
106
|
+
sm: 'sm:max-w-[400px]',
|
|
107
|
+
md: 'sm:max-w-[500px]',
|
|
108
|
+
lg: 'sm:max-w-[640px]',
|
|
109
|
+
xl: 'sm:max-w-[800px]'
|
|
110
|
+
};
|
|
111
|
+
const maxWidth = sizeMap[this.size] || sizeMap.lg;
|
|
112
|
+
return this.buildClasses([
|
|
113
|
+
// Mobile: fullscreen
|
|
114
|
+
'w-full',
|
|
115
|
+
'h-full',
|
|
116
|
+
'max-h-[100dvh]',
|
|
117
|
+
'rounded-t-[10px]',
|
|
118
|
+
// Desktop: modal
|
|
119
|
+
'sm:rounded-[10px]',
|
|
120
|
+
'sm:w-auto',
|
|
121
|
+
'sm:h-auto',
|
|
122
|
+
maxWidth,
|
|
123
|
+
'sm:max-h-[calc(100vh-4rem)]'
|
|
124
|
+
]);
|
|
125
|
+
}
|
|
126
|
+
buildClasses(sizeClasses) {
|
|
47
127
|
const base = [
|
|
48
|
-
'relative
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
128
|
+
'relative',
|
|
129
|
+
Z_INDEX.modal,
|
|
130
|
+
'flex',
|
|
131
|
+
'flex-col',
|
|
132
|
+
'border',
|
|
133
|
+
'border-border',
|
|
134
|
+
'bg-background',
|
|
135
|
+
'text-foreground',
|
|
136
|
+
'shadow-lg',
|
|
137
|
+
'overflow-hidden',
|
|
138
|
+
...sizeClasses,
|
|
54
139
|
this.className
|
|
55
140
|
];
|
|
56
141
|
return base.filter(Boolean).join(' ');
|
|
57
142
|
}
|
|
58
143
|
get bodyWrapperClassName() {
|
|
59
144
|
const base = [
|
|
60
|
-
'
|
|
61
|
-
|
|
145
|
+
'flex-1',
|
|
146
|
+
'overflow-y-auto',
|
|
147
|
+
'px-4',
|
|
148
|
+
'py-6',
|
|
149
|
+
'sm:px-6',
|
|
62
150
|
this.bodyClassName
|
|
63
151
|
];
|
|
64
152
|
return base.filter(Boolean).join(' ');
|
|
65
153
|
}
|
|
66
154
|
get headerWrapperClassName() {
|
|
67
|
-
|
|
155
|
+
const base = [
|
|
156
|
+
'flex',
|
|
157
|
+
'items-start',
|
|
158
|
+
'justify-between',
|
|
159
|
+
'gap-3',
|
|
160
|
+
'p-4',
|
|
161
|
+
'sm:p-6',
|
|
162
|
+
'border-b',
|
|
163
|
+
'border-border',
|
|
164
|
+
this.headerClassName
|
|
165
|
+
];
|
|
166
|
+
return base.filter(Boolean).join(' ');
|
|
68
167
|
}
|
|
69
168
|
get footerWrapperClassName() {
|
|
70
169
|
const effectiveAlign = this.alignFooter === 'right' && this.variant === 'custom-close' ? 'left' : this.alignFooter;
|
|
71
170
|
const base = [
|
|
72
171
|
'p-4',
|
|
172
|
+
'sm:p-6',
|
|
173
|
+
'border-t',
|
|
174
|
+
'border-border',
|
|
175
|
+
// Mobile: siempre full-width
|
|
176
|
+
'flex',
|
|
177
|
+
'flex-col',
|
|
178
|
+
'gap-2',
|
|
179
|
+
// Desktop: según alignFooter
|
|
73
180
|
effectiveAlign === 'full-width'
|
|
74
|
-
? 'flex
|
|
75
|
-
:
|
|
76
|
-
|
|
77
|
-
|
|
181
|
+
? 'sm:flex-col'
|
|
182
|
+
: 'sm:flex-row sm:items-center',
|
|
183
|
+
effectiveAlign === 'left' ? 'sm:justify-start' : '',
|
|
184
|
+
effectiveAlign === 'right' ? 'sm:justify-end' : '',
|
|
78
185
|
this.footerClassName
|
|
79
186
|
];
|
|
80
187
|
return base.filter(Boolean).join(' ');
|
|
81
188
|
}
|
|
189
|
+
get containerClassName() {
|
|
190
|
+
// Container con backdrop z-50
|
|
191
|
+
// Mobile: fullscreen desde el bottom
|
|
192
|
+
// Desktop: centrado
|
|
193
|
+
return responsive({
|
|
194
|
+
default: `fixed inset-x-0 bottom-0 ${Z_INDEX.modalBackdrop} flex items-end justify-center`,
|
|
195
|
+
sm: `fixed inset-0 ${Z_INDEX.modalBackdrop} flex items-center justify-center p-4`
|
|
196
|
+
});
|
|
197
|
+
}
|
|
82
198
|
}
|
|
83
199
|
PdmDialogComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: PdmDialogComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
84
|
-
PdmDialogComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: PdmDialogComponent, selector: "pdm-dialog", inputs: { open: "open", variant: "variant", size: "size", title: "title", description: "description", closeOnBackdrop: "closeOnBackdrop", closeOnEsc: "closeOnEsc", showCloseButton: "showCloseButton", showHeader: "showHeader", showFooter: "showFooter", primaryActionText: "primaryActionText", secondaryActionText: "secondaryActionText", alignFooter: "alignFooter", headerClassName: "headerClassName", bodyClassName: "bodyClassName", footerClassName: "footerClassName", className: "className" }, outputs: { openChange: "openChange", primaryAction: "primaryAction", secondaryAction: "secondaryAction" }, host: { listeners: { "document:keydown.escape": "onEsc()" } }, ngImport: i0, template: "<div *ngIf=\"open\"
|
|
200
|
+
PdmDialogComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: PdmDialogComponent, selector: "pdm-dialog", inputs: { open: "open", variant: "variant", size: "size", title: "title", description: "description", closeOnBackdrop: "closeOnBackdrop", closeOnEsc: "closeOnEsc", showCloseButton: "showCloseButton", showHeader: "showHeader", showFooter: "showFooter", primaryActionText: "primaryActionText", secondaryActionText: "secondaryActionText", alignFooter: "alignFooter", headerClassName: "headerClassName", bodyClassName: "bodyClassName", footerClassName: "footerClassName", className: "className" }, outputs: { openChange: "openChange", primaryAction: "primaryAction", secondaryAction: "secondaryAction" }, host: { listeners: { "document:keydown.escape": "onEsc()" } }, ngImport: i0, template: "<div *ngIf=\"open\" [ngClass]=\"containerClassName\">\n <!-- Backdrop -->\n <div class=\"absolute inset-0 bg-foreground/30 backdrop-blur-sm\" (click)=\"onBackdropClick()\"></div>\n \n <!-- Dialog Panel -->\n <section role=\"dialog\" aria-modal=\"true\" [ngClass]=\"panelClassName\">\n <!-- Header -->\n <div *ngIf=\"showHeader\" [ngClass]=\"headerWrapperClassName\">\n <div class=\"min-w-0 flex-1\">\n <h2 class=\"m-0 text-lg font-semibold leading-none tracking-tight text-foreground\">{{ title }}</h2>\n <p *ngIf=\"description\" class=\"m-0 mt-2 text-sm text-muted-foreground\">{{ description }}</p>\n </div>\n <button\n *ngIf=\"showCloseButton\"\n type=\"button\"\n class=\"inline-flex h-6 w-6 flex-shrink-0 appearance-none items-center justify-center rounded-sm border-0 bg-transparent p-0 text-foreground opacity-70 ring-offset-background transition-opacity hover:opacity-100 focus:outline-none focus:ring-2 focus:ring-ring focus:ring-offset-2 disabled:pointer-events-none\"\n (click)=\"close()\"\n aria-label=\"Close dialog\"\n >\n <svg viewBox=\"0 0 24 24\" class=\"h-4 w-4\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path d=\"M6 6L18 18M18 6L6 18\" stroke=\"currentColor\" stroke-width=\"1.5\" stroke-linecap=\"round\"></path>\n </svg>\n </button>\n </div>\n\n <!-- Body -->\n <div [ngClass]=\"bodyWrapperClassName\">\n <ng-content></ng-content>\n </div>\n\n <!-- Footer -->\n <div *ngIf=\"showFooter\" [ngClass]=\"footerWrapperClassName\">\n <ng-container *ngIf=\"variant === 'custom-close'; else defaultActions\">\n <button\n type=\"button\"\n class=\"inline-flex h-9 w-full appearance-none items-center justify-center rounded-md border border-input bg-background px-4 py-2 text-sm font-medium text-foreground shadow-sm sm:w-auto\"\n (click)=\"onSecondaryAction()\"\n >\n {{ secondaryActionText }}\n </button>\n </ng-container>\n\n <ng-template #defaultActions>\n <button\n type=\"button\"\n class=\"inline-flex h-9 w-full appearance-none items-center justify-center rounded-md border border-input bg-background px-4 py-2 text-sm font-medium text-foreground shadow-sm sm:w-auto\"\n (click)=\"onSecondaryAction()\"\n >\n {{ secondaryActionText }}\n </button>\n <button\n type=\"button\"\n class=\"inline-flex h-9 w-full appearance-none items-center justify-center rounded-md bg-primary px-4 py-2 text-sm font-medium text-primary-foreground shadow-sm sm:w-auto\"\n (click)=\"onPrimaryAction()\"\n >\n {{ primaryActionText }}\n </button>\n </ng-template>\n </div>\n </section>\n</div>\n", dependencies: [{ kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
85
201
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: PdmDialogComponent, decorators: [{
|
|
86
202
|
type: Component,
|
|
87
|
-
args: [{ selector: 'pdm-dialog', changeDetection: ChangeDetectionStrategy.OnPush, template: "<div *ngIf=\"open\"
|
|
203
|
+
args: [{ selector: 'pdm-dialog', changeDetection: ChangeDetectionStrategy.OnPush, template: "<div *ngIf=\"open\" [ngClass]=\"containerClassName\">\n <!-- Backdrop -->\n <div class=\"absolute inset-0 bg-foreground/30 backdrop-blur-sm\" (click)=\"onBackdropClick()\"></div>\n \n <!-- Dialog Panel -->\n <section role=\"dialog\" aria-modal=\"true\" [ngClass]=\"panelClassName\">\n <!-- Header -->\n <div *ngIf=\"showHeader\" [ngClass]=\"headerWrapperClassName\">\n <div class=\"min-w-0 flex-1\">\n <h2 class=\"m-0 text-lg font-semibold leading-none tracking-tight text-foreground\">{{ title }}</h2>\n <p *ngIf=\"description\" class=\"m-0 mt-2 text-sm text-muted-foreground\">{{ description }}</p>\n </div>\n <button\n *ngIf=\"showCloseButton\"\n type=\"button\"\n class=\"inline-flex h-6 w-6 flex-shrink-0 appearance-none items-center justify-center rounded-sm border-0 bg-transparent p-0 text-foreground opacity-70 ring-offset-background transition-opacity hover:opacity-100 focus:outline-none focus:ring-2 focus:ring-ring focus:ring-offset-2 disabled:pointer-events-none\"\n (click)=\"close()\"\n aria-label=\"Close dialog\"\n >\n <svg viewBox=\"0 0 24 24\" class=\"h-4 w-4\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n <path d=\"M6 6L18 18M18 6L6 18\" stroke=\"currentColor\" stroke-width=\"1.5\" stroke-linecap=\"round\"></path>\n </svg>\n </button>\n </div>\n\n <!-- Body -->\n <div [ngClass]=\"bodyWrapperClassName\">\n <ng-content></ng-content>\n </div>\n\n <!-- Footer -->\n <div *ngIf=\"showFooter\" [ngClass]=\"footerWrapperClassName\">\n <ng-container *ngIf=\"variant === 'custom-close'; else defaultActions\">\n <button\n type=\"button\"\n class=\"inline-flex h-9 w-full appearance-none items-center justify-center rounded-md border border-input bg-background px-4 py-2 text-sm font-medium text-foreground shadow-sm sm:w-auto\"\n (click)=\"onSecondaryAction()\"\n >\n {{ secondaryActionText }}\n </button>\n </ng-container>\n\n <ng-template #defaultActions>\n <button\n type=\"button\"\n class=\"inline-flex h-9 w-full appearance-none items-center justify-center rounded-md border border-input bg-background px-4 py-2 text-sm font-medium text-foreground shadow-sm sm:w-auto\"\n (click)=\"onSecondaryAction()\"\n >\n {{ secondaryActionText }}\n </button>\n <button\n type=\"button\"\n class=\"inline-flex h-9 w-full appearance-none items-center justify-center rounded-md bg-primary px-4 py-2 text-sm font-medium text-primary-foreground shadow-sm sm:w-auto\"\n (click)=\"onPrimaryAction()\"\n >\n {{ primaryActionText }}\n </button>\n </ng-template>\n </div>\n </section>\n</div>\n" }]
|
|
88
204
|
}], propDecorators: { open: [{
|
|
89
205
|
type: Input
|
|
90
206
|
}], variant: [{
|
|
@@ -129,4 +245,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImpor
|
|
|
129
245
|
type: HostListener,
|
|
130
246
|
args: ['document:keydown.escape']
|
|
131
247
|
}] } });
|
|
132
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dialog.component.js","sourceRoot":"","sources":["../../../../../../src/lib/components/dialog/dialog.component.ts","../../../../../../src/lib/components/dialog/dialog.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, Component, EventEmitter, HostListener, Input, Output } from '@angular/core';\n\nexport type PdmDialogVariant = 'default' | 'custom-close';\nexport type PdmDialogSize = 'desktop' | 'mobile' | 'mobile-fullscreen';\nexport type PdmDialogFooterAlign = 'right' | 'full-width' | 'left';\n\n@Component({\n  selector: 'pdm-dialog',\n  templateUrl: './dialog.component.html',\n  changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class PdmDialogComponent {\n  @Input() open = false;\n  @Input() variant: PdmDialogVariant = 'default';\n  @Input() size: PdmDialogSize = 'desktop';\n  @Input() title = 'Edit profile';\n  @Input() description = '';\n  @Input() closeOnBackdrop = true;\n  @Input() closeOnEsc = true;\n  @Input() showCloseButton = true;\n  @Input() showHeader = true;\n  @Input() showFooter = true;\n  @Input() primaryActionText = 'Save changes';\n  @Input() secondaryActionText = 'Cancel';\n  @Input() alignFooter: PdmDialogFooterAlign = 'right';\n  @Input() headerClassName = '';\n  @Input() bodyClassName = '';\n  @Input() footerClassName = '';\n  @Input() className = '';\n\n  @Output() openChange = new EventEmitter<boolean>();\n  @Output() primaryAction = new EventEmitter<void>();\n  @Output() secondaryAction = new EventEmitter<void>();\n\n  @HostListener('document:keydown.escape')\n  onEsc(): void {\n    if (this.open && this.closeOnEsc) {\n      this.close();\n    }\n  }\n\n  close(): void {\n    this.openChange.emit(false);\n  }\n\n  onPrimaryAction(): void {\n    this.primaryAction.emit();\n  }\n\n  onSecondaryAction(): void {\n    this.secondaryAction.emit();\n  }\n\n  onBackdropClick(): void {\n    if (this.closeOnBackdrop) {\n      this.close();\n    }\n  }\n\n  get panelClassName(): string {\n    const base = [\n      'relative z-10 w-full border border-border bg-background text-foreground shadow-lg',\n      this.size === 'desktop' ? 'max-w-[640px] max-h-[calc(100vh-2rem)] rounded-[10px] overflow-visible' : '',\n      this.size === 'mobile' ? 'max-w-[320px] min-h-[240px] rounded-[10px] overflow-visible' : '',\n      this.size === 'mobile-fullscreen'\n        ? 'max-w-[320px] h-[min(100dvh,640px)] rounded-none sm:rounded-[10px] overflow-visible'\n        : '',\n      this.className\n    ];\n\n    return base.filter(Boolean).join(' ');\n  }\n\n  get bodyWrapperClassName(): string {\n    const base = [\n      'min-h-0 flex-1',\n      this.size === 'mobile-fullscreen' ? 'overflow-y-auto px-4 py-6' : 'px-6 py-6',\n      this.bodyClassName\n    ];\n\n    return base.filter(Boolean).join(' ');\n  }\n\n  get headerWrapperClassName(): string {\n    return ['flex items-start justify-between gap-3 p-4', this.headerClassName].filter(Boolean).join(' ');\n  }\n\n  get footerWrapperClassName(): string {\n    const effectiveAlign =\n      this.alignFooter === 'right' && this.variant === 'custom-close' ? 'left' : this.alignFooter;\n\n    const base = [\n      'p-4',\n      effectiveAlign === 'full-width'\n        ? 'flex flex-col gap-2'\n        : effectiveAlign === 'left'\n          ? 'flex items-center gap-2 justify-start'\n          : 'flex items-center gap-2 justify-end',\n      this.footerClassName\n    ];\n\n    return base.filter(Boolean).join(' ');\n  }\n}\n","<div *ngIf=\"open\" class=\"fixed inset-0 z-50 flex items-center justify-center p-4\">\n  <div class=\"absolute inset-0 bg-foreground/30\" (click)=\"onBackdropClick()\"></div>\n  <section role=\"dialog\" aria-modal=\"true\" [ngClass]=\"panelClassName\">\n    <div *ngIf=\"showHeader\" [ngClass]=\"headerWrapperClassName\">\n      <div class=\"min-w-0\">\n        <h2 class=\"m-0 text-lg font-semibold leading-none tracking-tight text-foreground\">{{ title }}</h2>\n        <p *ngIf=\"description\" class=\"m-0 mt-2 text-sm text-muted-foreground\">{{ description }}</p>\n      </div>\n      <button\n        *ngIf=\"showCloseButton\"\n        type=\"button\"\n        class=\"inline-flex h-6 w-6 appearance-none items-center justify-center rounded-sm border-0 bg-transparent p-0 text-foreground opacity-70 ring-offset-background transition-opacity hover:opacity-100 focus:outline-none focus:ring-2 focus:ring-ring focus:ring-offset-2 disabled:pointer-events-none\"\n        (click)=\"close()\"\n        aria-label=\"Close dialog\"\n      >\n        <svg viewBox=\"0 0 24 24\" class=\"h-4 w-4\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n          <path d=\"M6 6L18 18M18 6L6 18\" stroke=\"currentColor\" stroke-width=\"1.5\" stroke-linecap=\"round\"></path>\n        </svg>\n      </button>\n    </div>\n\n    <div [ngClass]=\"bodyWrapperClassName\">\n      <ng-content></ng-content>\n    </div>\n\n    <div *ngIf=\"showFooter\" [ngClass]=\"footerWrapperClassName\">\n      <ng-container *ngIf=\"variant === 'custom-close'; else defaultActions\">\n        <button\n          type=\"button\"\n          class=\"inline-flex h-9 appearance-none items-center justify-center rounded-md border border-input bg-background px-4 py-2 text-sm font-medium text-foreground shadow-sm\"\n          (click)=\"onSecondaryAction()\"\n        >\n          {{ secondaryActionText }}\n        </button>\n      </ng-container>\n\n      <ng-template #defaultActions>\n        <button\n          type=\"button\"\n          class=\"inline-flex h-9 appearance-none items-center justify-center rounded-md border border-input bg-background px-4 py-2 text-sm font-medium text-foreground shadow-sm\"\n          (click)=\"onSecondaryAction()\"\n        >\n          {{ secondaryActionText }}\n        </button>\n        <button\n          type=\"button\"\n          class=\"inline-flex h-9 appearance-none items-center justify-center rounded-md bg-primary px-4 py-2 text-sm font-medium text-primary-foreground shadow-sm\"\n          (click)=\"onPrimaryAction()\"\n        >\n          {{ primaryActionText }}\n        </button>\n      </ng-template>\n    </div>\n  </section>\n</div>\n"]}
|
|
248
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dialog.component.js","sourceRoot":"","sources":["../../../../../../src/lib/components/dialog/dialog.component.ts","../../../../../../src/lib/components/dialog/dialog.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, Component, EventEmitter, HostListener, Input, Output } from '@angular/core';\nimport { responsive } from '../../utils/responsive';\nimport { Z_INDEX } from '../../utils/z-index';\n\nexport type PdmDialogVariant = 'default' | 'custom-close';\n\n/**\n * @deprecated Use 'responsive' mode instead. Will be removed in v0.3.0\n */\nexport type PdmDialogSize = 'desktop' | 'mobile' | 'mobile-fullscreen' | 'sm' | 'md' | 'lg' | 'xl' | 'responsive';\n\nexport type PdmDialogFooterAlign = 'right' | 'full-width' | 'left';\n\n/**\n * Modal/Dialog component con soporte responsive\n * \n * MEJORADO en v0.2.0:\n * - Modo 'responsive' (default): fullscreen en mobile, modal en desktop\n * - Tamaños predefinidos: sm, md, lg, xl\n * - Mejor manejo de scroll en mobile\n * \n * @example\n * // Responsive (recomendado)\n * <pdm-dialog [open]=\"isOpen\" size=\"responsive\">\n *   <p>Content</p>\n * </pdm-dialog>\n * \n * @example\n * // Tamaño fijo\n * <pdm-dialog [open]=\"isOpen\" size=\"lg\">\n *   <p>Content</p>\n * </pdm-dialog>\n */\n@Component({\n  selector: 'pdm-dialog',\n  templateUrl: './dialog.component.html',\n  changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class PdmDialogComponent {\n  @Input() open = false;\n  @Input() variant: PdmDialogVariant = 'default';\n  \n  /**\n   * Tamaño del dialog\n   * - responsive: fullscreen mobile, modal desktop (recomendado)\n   * - sm: 400px max\n   * - md: 500px max\n   * - lg: 640px max (default)\n   * - xl: 800px max\n   * - desktop/mobile/mobile-fullscreen: legacy, deprecado\n   */\n  @Input() size: PdmDialogSize = 'responsive';\n  \n  @Input() title = 'Edit profile';\n  @Input() description = '';\n  @Input() closeOnBackdrop = true;\n  @Input() closeOnEsc = true;\n  @Input() showCloseButton = true;\n  @Input() showHeader = true;\n  @Input() showFooter = true;\n  @Input() primaryActionText = 'Save changes';\n  @Input() secondaryActionText = 'Cancel';\n  @Input() alignFooter: PdmDialogFooterAlign = 'right';\n  @Input() headerClassName = '';\n  @Input() bodyClassName = '';\n  @Input() footerClassName = '';\n  @Input() className = '';\n\n  @Output() openChange = new EventEmitter<boolean>();\n  @Output() primaryAction = new EventEmitter<void>();\n  @Output() secondaryAction = new EventEmitter<void>();\n\n  @HostListener('document:keydown.escape')\n  onEsc(): void {\n    if (this.open && this.closeOnEsc) {\n      this.close();\n    }\n  }\n\n  close(): void {\n    this.openChange.emit(false);\n  }\n\n  onPrimaryAction(): void {\n    this.primaryAction.emit();\n  }\n\n  onSecondaryAction(): void {\n    this.secondaryAction.emit();\n  }\n\n  onBackdropClick(): void {\n    if (this.closeOnBackdrop) {\n      this.close();\n    }\n  }\n\n  get panelClassName(): string {\n    // Legacy sizes (backward compatibility)\n    if (this.size === 'desktop') {\n      return this.buildClasses(['max-w-[640px]', 'max-h-[calc(100vh-2rem)]', 'rounded-[10px]']);\n    }\n    \n    if (this.size === 'mobile') {\n      return this.buildClasses(['max-w-[320px]', 'min-h-[240px]', 'rounded-[10px]']);\n    }\n    \n    if (this.size === 'mobile-fullscreen') {\n      return this.buildClasses(['max-w-[320px]', 'h-[min(100dvh,640px)]', 'rounded-none', 'sm:rounded-[10px]']);\n    }\n\n    // New responsive mode (recomendado)\n    if (this.size === 'responsive') {\n      return this.buildClasses([\n        // Mobile: fullscreen con bordes redondeados solo arriba\n        'w-full',\n        'h-full',\n        'max-h-[100dvh]',\n        'rounded-t-[10px]',\n        'sm:rounded-[10px]',\n        // Desktop: modal centrado\n        'sm:w-auto',\n        'sm:h-auto',\n        'sm:max-w-[640px]',\n        'sm:max-h-[calc(100vh-4rem)]'\n      ]);\n    }\n\n    // New size options\n    const sizeMap = {\n      sm: 'sm:max-w-[400px]',\n      md: 'sm:max-w-[500px]',\n      lg: 'sm:max-w-[640px]',\n      xl: 'sm:max-w-[800px]'\n    };\n\n    const maxWidth = sizeMap[this.size as keyof typeof sizeMap] || sizeMap.lg;\n\n    return this.buildClasses([\n      // Mobile: fullscreen\n      'w-full',\n      'h-full',\n      'max-h-[100dvh]',\n      'rounded-t-[10px]',\n      // Desktop: modal\n      'sm:rounded-[10px]',\n      'sm:w-auto',\n      'sm:h-auto',\n      maxWidth,\n      'sm:max-h-[calc(100vh-4rem)]'\n    ]);\n  }\n\n  private buildClasses(sizeClasses: string[]): string {\n    const base = [\n      'relative',\n      Z_INDEX.modal, // z-[60] - debe estar sobre backdrop (z-50)\n      'flex',\n      'flex-col',\n      'border',\n      'border-border',\n      'bg-background',\n      'text-foreground',\n      'shadow-lg',\n      'overflow-hidden',\n      ...sizeClasses,\n      this.className\n    ];\n\n    return base.filter(Boolean).join(' ');\n  }\n\n  get bodyWrapperClassName(): string {\n    const base = [\n      'flex-1',\n      'overflow-y-auto',\n      'px-4',\n      'py-6',\n      'sm:px-6',\n      this.bodyClassName\n    ];\n\n    return base.filter(Boolean).join(' ');\n  }\n\n  get headerWrapperClassName(): string {\n    const base = [\n      'flex',\n      'items-start',\n      'justify-between',\n      'gap-3',\n      'p-4',\n      'sm:p-6',\n      'border-b',\n      'border-border',\n      this.headerClassName\n    ];\n    \n    return base.filter(Boolean).join(' ');\n  }\n\n  get footerWrapperClassName(): string {\n    const effectiveAlign =\n      this.alignFooter === 'right' && this.variant === 'custom-close' ? 'left' : this.alignFooter;\n\n    const base = [\n      'p-4',\n      'sm:p-6',\n      'border-t',\n      'border-border',\n      // Mobile: siempre full-width\n      'flex',\n      'flex-col',\n      'gap-2',\n      // Desktop: según alignFooter\n      effectiveAlign === 'full-width'\n        ? 'sm:flex-col'\n        : 'sm:flex-row sm:items-center',\n      effectiveAlign === 'left' ? 'sm:justify-start' : '',\n      effectiveAlign === 'right' ? 'sm:justify-end' : '',\n      this.footerClassName\n    ];\n\n    return base.filter(Boolean).join(' ');\n  }\n\n  get containerClassName(): string {\n    // Container con backdrop z-50\n    // Mobile: fullscreen desde el bottom\n    // Desktop: centrado\n    return responsive({\n      default: `fixed inset-x-0 bottom-0 ${Z_INDEX.modalBackdrop} flex items-end justify-center`,\n      sm: `fixed inset-0 ${Z_INDEX.modalBackdrop} flex items-center justify-center p-4`\n    });\n  }\n}\n","<div *ngIf=\"open\" [ngClass]=\"containerClassName\">\n  <!-- Backdrop -->\n  <div class=\"absolute inset-0 bg-foreground/30 backdrop-blur-sm\" (click)=\"onBackdropClick()\"></div>\n  \n  <!-- Dialog Panel -->\n  <section role=\"dialog\" aria-modal=\"true\" [ngClass]=\"panelClassName\">\n    <!-- Header -->\n    <div *ngIf=\"showHeader\" [ngClass]=\"headerWrapperClassName\">\n      <div class=\"min-w-0 flex-1\">\n        <h2 class=\"m-0 text-lg font-semibold leading-none tracking-tight text-foreground\">{{ title }}</h2>\n        <p *ngIf=\"description\" class=\"m-0 mt-2 text-sm text-muted-foreground\">{{ description }}</p>\n      </div>\n      <button\n        *ngIf=\"showCloseButton\"\n        type=\"button\"\n        class=\"inline-flex h-6 w-6 flex-shrink-0 appearance-none items-center justify-center rounded-sm border-0 bg-transparent p-0 text-foreground opacity-70 ring-offset-background transition-opacity hover:opacity-100 focus:outline-none focus:ring-2 focus:ring-ring focus:ring-offset-2 disabled:pointer-events-none\"\n        (click)=\"close()\"\n        aria-label=\"Close dialog\"\n      >\n        <svg viewBox=\"0 0 24 24\" class=\"h-4 w-4\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n          <path d=\"M6 6L18 18M18 6L6 18\" stroke=\"currentColor\" stroke-width=\"1.5\" stroke-linecap=\"round\"></path>\n        </svg>\n      </button>\n    </div>\n\n    <!-- Body -->\n    <div [ngClass]=\"bodyWrapperClassName\">\n      <ng-content></ng-content>\n    </div>\n\n    <!-- Footer -->\n    <div *ngIf=\"showFooter\" [ngClass]=\"footerWrapperClassName\">\n      <ng-container *ngIf=\"variant === 'custom-close'; else defaultActions\">\n        <button\n          type=\"button\"\n          class=\"inline-flex h-9 w-full appearance-none items-center justify-center rounded-md border border-input bg-background px-4 py-2 text-sm font-medium text-foreground shadow-sm sm:w-auto\"\n          (click)=\"onSecondaryAction()\"\n        >\n          {{ secondaryActionText }}\n        </button>\n      </ng-container>\n\n      <ng-template #defaultActions>\n        <button\n          type=\"button\"\n          class=\"inline-flex h-9 w-full appearance-none items-center justify-center rounded-md border border-input bg-background px-4 py-2 text-sm font-medium text-foreground shadow-sm sm:w-auto\"\n          (click)=\"onSecondaryAction()\"\n        >\n          {{ secondaryActionText }}\n        </button>\n        <button\n          type=\"button\"\n          class=\"inline-flex h-9 w-full appearance-none items-center justify-center rounded-md bg-primary px-4 py-2 text-sm font-medium text-primary-foreground shadow-sm sm:w-auto\"\n          (click)=\"onPrimaryAction()\"\n        >\n          {{ primaryActionText }}\n        </button>\n      </ng-template>\n    </div>\n  </section>\n</div>\n"]}
|