valtech-components 2.0.628 → 2.0.630

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 (25) hide show
  1. package/esm2022/lib/components/molecules/action-card/action-card.component.mjs +298 -0
  2. package/esm2022/lib/components/molecules/action-card/types.mjs +11 -0
  3. package/esm2022/lib/components/molecules/linked-providers/linked-providers.component.mjs +236 -0
  4. package/esm2022/lib/components/molecules/linked-providers/types.mjs +27 -0
  5. package/esm2022/lib/components/molecules/stats-card/stats-card.component.mjs +33 -3
  6. package/esm2022/lib/components/molecules/stats-card/types.mjs +1 -1
  7. package/esm2022/lib/components/molecules/username-input/types.mjs +2 -0
  8. package/esm2022/lib/components/molecules/username-input/username-input.component.mjs +260 -0
  9. package/esm2022/lib/components/templates/docs-page/docs-page.component.mjs +3 -3
  10. package/esm2022/lib/services/auth/auth.service.mjs +24 -1
  11. package/esm2022/lib/services/auth/types.mjs +1 -1
  12. package/esm2022/public-api.mjs +7 -1
  13. package/fesm2022/valtech-components.mjs +904 -46
  14. package/fesm2022/valtech-components.mjs.map +1 -1
  15. package/lib/components/molecules/action-card/action-card.component.d.ts +90 -0
  16. package/lib/components/molecules/action-card/types.d.ts +83 -0
  17. package/lib/components/molecules/linked-providers/linked-providers.component.d.ts +30 -0
  18. package/lib/components/molecules/linked-providers/types.d.ts +38 -0
  19. package/lib/components/molecules/stats-card/types.d.ts +17 -0
  20. package/lib/components/molecules/username-input/types.d.ts +34 -0
  21. package/lib/components/molecules/username-input/username-input.component.d.ts +45 -0
  22. package/lib/services/auth/auth.service.d.ts +11 -1
  23. package/lib/services/auth/types.d.ts +56 -0
  24. package/package.json +1 -1
  25. package/public-api.d.ts +6 -0
@@ -0,0 +1,27 @@
1
+ /**
2
+ * Información de proveedores OAuth disponibles.
3
+ */
4
+ export const OAUTH_PROVIDERS_INFO = {
5
+ google: {
6
+ id: 'google',
7
+ name: 'Google',
8
+ icon: 'logo-google',
9
+ color: '#DB4437',
10
+ bgColor: '#DB443715',
11
+ },
12
+ apple: {
13
+ id: 'apple',
14
+ name: 'Apple',
15
+ icon: 'logo-apple',
16
+ color: '#000000',
17
+ bgColor: '#00000010',
18
+ },
19
+ microsoft: {
20
+ id: 'microsoft',
21
+ name: 'Microsoft',
22
+ icon: 'logo-microsoft',
23
+ color: '#00A4EF',
24
+ bgColor: '#00A4EF15',
25
+ },
26
+ };
27
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL2NvbXBvbmVudHMvbW9sZWN1bGVzL2xpbmtlZC1wcm92aWRlcnMvdHlwZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBcUNBOztHQUVHO0FBQ0gsTUFBTSxDQUFDLE1BQU0sb0JBQW9CLEdBQStDO0lBQzlFLE1BQU0sRUFBRTtRQUNOLEVBQUUsRUFBRSxRQUFRO1FBQ1osSUFBSSxFQUFFLFFBQVE7UUFDZCxJQUFJLEVBQUUsYUFBYTtRQUNuQixLQUFLLEVBQUUsU0FBUztRQUNoQixPQUFPLEVBQUUsV0FBVztLQUNyQjtJQUNELEtBQUssRUFBRTtRQUNMLEVBQUUsRUFBRSxPQUFPO1FBQ1gsSUFBSSxFQUFFLE9BQU87UUFDYixJQUFJLEVBQUUsWUFBWTtRQUNsQixLQUFLLEVBQUUsU0FBUztRQUNoQixPQUFPLEVBQUUsV0FBVztLQUNyQjtJQUNELFNBQVMsRUFBRTtRQUNULEVBQUUsRUFBRSxXQUFXO1FBQ2YsSUFBSSxFQUFFLFdBQVc7UUFDakIsSUFBSSxFQUFFLGdCQUFnQjtRQUN0QixLQUFLLEVBQUUsU0FBUztRQUNoQixPQUFPLEVBQUUsV0FBVztLQUNyQjtDQUNGLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBMaW5rZWRQcm92aWRlciwgT0F1dGhQcm92aWRlciB9IGZyb20gJy4uLy4uLy4uL3NlcnZpY2VzL2F1dGgvdHlwZXMnO1xuXG4vKipcbiAqIE1ldGFkYXRhIHBhcmEgTGlua2VkUHJvdmlkZXJzQ29tcG9uZW50LlxuICovXG5leHBvcnQgaW50ZXJmYWNlIExpbmtlZFByb3ZpZGVyc01ldGFkYXRhIHtcbiAgLyoqIExpc3RhIGRlIHByb3ZlZWRvcmVzIHZpbmN1bGFkb3MgKi9cbiAgcHJvdmlkZXJzOiBMaW5rZWRQcm92aWRlcltdO1xuICAvKiogUHJvdmVlZG9yZXMgZGlzcG9uaWJsZXMgcGFyYSB2aW5jdWxhciAoZGVmYXVsdDogWydnb29nbGUnXSkgKi9cbiAgYXZhaWxhYmxlUHJvdmlkZXJzPzogT0F1dGhQcm92aWRlcltdO1xuICAvKiogQ2FsbGJhY2sgY3VhbmRvIHNlIHF1aWVyZSB2aW5jdWxhciB1biBwcm92aWRlciAqL1xuICBvbkxpbms/OiAocHJvdmlkZXI6IE9BdXRoUHJvdmlkZXIpID0+IHZvaWQ7XG4gIC8qKiBDYWxsYmFjayBjdWFuZG8gc2UgcXVpZXJlIGRlc3ZpbmN1bGFyIHVuIHByb3ZpZGVyICovXG4gIG9uVW5saW5rPzogKHByb3ZpZGVyOiBPQXV0aFByb3ZpZGVyKSA9PiB2b2lkO1xuICAvKiogTW9zdHJhciBib3TDs24gcGFyYSB2aW5jdWxhciBudWV2b3MgKGRlZmF1bHQ6IHRydWUpICovXG4gIHNob3dMaW5rQnV0dG9uPzogYm9vbGVhbjtcbiAgLyoqIFBlcm1pdGlyIGRlc3ZpbmN1bGFyIChkZWZhdWx0OiB0cnVlIHNpIGhheSBtw6FzIGRlIHVuIG3DqXRvZG8gZGUgYXV0aCkgKi9cbiAgYWxsb3dVbmxpbms/OiBib29sZWFuO1xuICAvKiogVMOtdHVsbyBkZSBsYSBzZWNjacOzbiAqL1xuICB0aXRsZT86IHN0cmluZztcbiAgLyoqIERlc2NyaXBjacOzbiBkZSBsYSBzZWNjacOzbiAqL1xuICBkZXNjcmlwdGlvbj86IHN0cmluZztcbiAgLyoqIE1vZG8gY29tcGFjdG8gc2luIGNhcmQgKi9cbiAgY29tcGFjdD86IGJvb2xlYW47XG59XG5cbi8qKlxuICogSW5mb3JtYWNpw7NuIHZpc3VhbCBkZSB1biBwcm92ZWVkb3IgT0F1dGguXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgUHJvdmlkZXJEaXNwbGF5SW5mbyB7XG4gIGlkOiBPQXV0aFByb3ZpZGVyO1xuICBuYW1lOiBzdHJpbmc7XG4gIGljb246IHN0cmluZztcbiAgY29sb3I6IHN0cmluZztcbiAgYmdDb2xvcjogc3RyaW5nO1xufVxuXG4vKipcbiAqIEluZm9ybWFjacOzbiBkZSBwcm92ZWVkb3JlcyBPQXV0aCBkaXNwb25pYmxlcy5cbiAqL1xuZXhwb3J0IGNvbnN0IE9BVVRIX1BST1ZJREVSU19JTkZPOiBSZWNvcmQ8T0F1dGhQcm92aWRlciwgUHJvdmlkZXJEaXNwbGF5SW5mbz4gPSB7XG4gIGdvb2dsZToge1xuICAgIGlkOiAnZ29vZ2xlJyxcbiAgICBuYW1lOiAnR29vZ2xlJyxcbiAgICBpY29uOiAnbG9nby1nb29nbGUnLFxuICAgIGNvbG9yOiAnI0RCNDQzNycsXG4gICAgYmdDb2xvcjogJyNEQjQ0MzcxNScsXG4gIH0sXG4gIGFwcGxlOiB7XG4gICAgaWQ6ICdhcHBsZScsXG4gICAgbmFtZTogJ0FwcGxlJyxcbiAgICBpY29uOiAnbG9nby1hcHBsZScsXG4gICAgY29sb3I6ICcjMDAwMDAwJyxcbiAgICBiZ0NvbG9yOiAnIzAwMDAwMDEwJyxcbiAgfSxcbiAgbWljcm9zb2Z0OiB7XG4gICAgaWQ6ICdtaWNyb3NvZnQnLFxuICAgIG5hbWU6ICdNaWNyb3NvZnQnLFxuICAgIGljb246ICdsb2dvLW1pY3Jvc29mdCcsXG4gICAgY29sb3I6ICcjMDBBNEVGJyxcbiAgICBiZ0NvbG9yOiAnIzAwQTRFRjE1JyxcbiAgfSxcbn07XG4iXX0=
@@ -120,9 +120,24 @@ export class StatsCardComponent {
120
120
  @if (resolvedProps.footer && !resolvedProps.loading) {
121
121
  <div class="stats-footer">{{ resolvedProps.footer }}</div>
122
122
  }
123
+
124
+ @if (resolvedProps.description && !resolvedProps.loading) {
125
+ <div class="stats-description">{{ resolvedProps.description }}</div>
126
+ }
127
+
128
+ @if (resolvedProps.logo && !resolvedProps.loading) {
129
+ <div class="stats-logo">
130
+ <img
131
+ [src]="resolvedProps.logo.src"
132
+ [alt]="resolvedProps.logo.alt"
133
+ [style.max-height.px]="resolvedProps.logo.maxHeight || 32"
134
+ [style.max-width.px]="resolvedProps.logo.maxWidth || 120"
135
+ />
136
+ </div>
137
+ }
123
138
  </ion-card-content>
124
139
  </ion-card>
125
- `, isInline: true, styles: [":host{display:block}.stats-card{margin:0;border-radius:12px;cursor:pointer;transition:transform .2s,box-shadow .2s}.stats-card:hover{transform:translateY(-2px);box-shadow:0 8px 24px #0000001f}.stats-card.background-light{--background: var(--ion-color-light)}.stats-card.background-light .stats-title,.stats-card.background-light .stats-footer{color:var(--ion-color-medium)}.stats-card.background-light .stats-value{color:var(--ion-text-color)}.stats-card.background-light .stats-icon{color:var(--card-color);opacity:.8}.stats-card.background-solid{--background: var(--card-color)}.stats-card.background-solid .stats-title,.stats-card.background-solid .stats-footer,.stats-card.background-solid .stats-value,.stats-card.background-solid .stats-icon{color:#fff}.stats-card.background-solid .stats-title,.stats-card.background-solid .stats-footer{opacity:.9}.stats-card.background-gradient{--background: linear-gradient(135deg, var(--card-color), var(--card-color-shade, var(--card-color)))}.stats-card.background-gradient .stats-title,.stats-card.background-gradient .stats-footer,.stats-card.background-gradient .stats-value,.stats-card.background-gradient .stats-icon{color:#fff}.stats-card.background-gradient .stats-title,.stats-card.background-gradient .stats-footer{opacity:.9}ion-card-content{padding:16px}.stats-header{display:flex;justify-content:space-between;align-items:flex-start;margin-bottom:8px}.stats-title{font-size:13px;font-weight:500;text-transform:uppercase;letter-spacing:.5px}.stats-icon{font-size:24px}.stats-value{font-size:28px;font-weight:700;line-height:1.2;margin-bottom:8px}.stats-value .prefix,.stats-value .suffix{font-size:18px;font-weight:500;opacity:.8}.stats-value .prefix{margin-right:2px}.stats-value .suffix{margin-left:4px}.stats-trend{display:inline-flex;align-items:center;gap:4px;font-size:13px;font-weight:500;padding:4px 8px;border-radius:12px;margin-bottom:8px}.stats-trend ion-icon{font-size:16px}.stats-trend.trend-up{background-color:rgba(var(--ion-color-success-rgb),.15);color:var(--ion-color-success)}.stats-trend.trend-down{background-color:rgba(var(--ion-color-danger-rgb),.15);color:var(--ion-color-danger)}.stats-trend.trend-neutral{background-color:rgba(var(--ion-color-medium-rgb),.15);color:var(--ion-color-medium)}.stats-trend .trend-label{font-weight:400;opacity:.8;margin-left:4px}.stats-footer{font-size:12px;margin-top:4px}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "component", type: IonCard, selector: "ion-card", inputs: ["button", "color", "disabled", "download", "href", "mode", "rel", "routerAnimation", "routerDirection", "target", "type"] }, { kind: "component", type: IonCardContent, selector: "ion-card-content", inputs: ["mode"] }, { kind: "component", type: IonIcon, selector: "ion-icon", inputs: ["color", "flipRtl", "icon", "ios", "lazy", "md", "mode", "name", "sanitize", "size", "src"] }, { kind: "component", type: IonSkeletonText, selector: "ion-skeleton-text", inputs: ["animated"] }] }); }
140
+ `, isInline: true, styles: [":host{display:block}.stats-card{margin:0;border-radius:12px;cursor:pointer;transition:transform .2s,box-shadow .2s}.stats-card:hover{transform:translateY(-2px);box-shadow:0 8px 24px #0000001f}.stats-card.background-light{--background: var(--ion-color-light)}.stats-card.background-light .stats-title,.stats-card.background-light .stats-footer{color:var(--ion-color-medium)}.stats-card.background-light .stats-value{color:var(--ion-text-color)}.stats-card.background-light .stats-icon{color:var(--card-color);opacity:.8}.stats-card.background-solid{--background: var(--card-color)}.stats-card.background-solid .stats-title,.stats-card.background-solid .stats-footer,.stats-card.background-solid .stats-value,.stats-card.background-solid .stats-icon{color:#fff}.stats-card.background-solid .stats-title,.stats-card.background-solid .stats-footer{opacity:.9}.stats-card.background-gradient{--background: linear-gradient(135deg, var(--card-color), var(--card-color-shade, var(--card-color)))}.stats-card.background-gradient .stats-title,.stats-card.background-gradient .stats-footer,.stats-card.background-gradient .stats-value,.stats-card.background-gradient .stats-icon{color:#fff}.stats-card.background-gradient .stats-title,.stats-card.background-gradient .stats-footer{opacity:.9}ion-card-content{padding:16px}.stats-header{display:flex;justify-content:space-between;align-items:flex-start;margin-bottom:8px}.stats-title{font-size:13px;font-weight:500;text-transform:uppercase;letter-spacing:.5px}.stats-icon{font-size:24px}.stats-value{font-size:28px;font-weight:700;line-height:1.2;margin-bottom:8px}.stats-value .prefix,.stats-value .suffix{font-size:18px;font-weight:500;opacity:.8}.stats-value .prefix{margin-right:2px}.stats-value .suffix{margin-left:4px}.stats-trend{display:inline-flex;align-items:center;gap:4px;font-size:13px;font-weight:500;padding:4px 8px;border-radius:12px;margin-bottom:8px}.stats-trend ion-icon{font-size:16px}.stats-trend.trend-up{background-color:rgba(var(--ion-color-success-rgb),.15);color:var(--ion-color-success)}.stats-trend.trend-down{background-color:rgba(var(--ion-color-danger-rgb),.15);color:var(--ion-color-danger)}.stats-trend.trend-neutral{background-color:rgba(var(--ion-color-medium-rgb),.15);color:var(--ion-color-medium)}.stats-trend .trend-label{font-weight:400;opacity:.8;margin-left:4px}.stats-footer{font-size:12px;margin-top:4px}.stats-description{margin-top:8px;font-size:14px;line-height:1.4;color:var(--ion-text-color);opacity:.8}.stats-logo{margin-top:16px;padding-top:12px;border-top:1px solid rgba(var(--ion-text-color-rgb),.1)}.stats-logo img{display:block;object-fit:contain;filter:grayscale(100%);opacity:.7;transition:filter .2s,opacity .2s}.stats-logo:hover img{filter:grayscale(0%);opacity:1}.background-solid .stats-description,.background-gradient .stats-description{color:#fff;opacity:.9}.background-solid .stats-logo,.background-gradient .stats-logo{border-top-color:#fff3}.background-solid .stats-logo img,.background-gradient .stats-logo img{filter:grayscale(100%) brightness(10)}.background-solid .stats-logo:hover img,.background-gradient .stats-logo:hover img{filter:brightness(10)}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "component", type: IonCard, selector: "ion-card", inputs: ["button", "color", "disabled", "download", "href", "mode", "rel", "routerAnimation", "routerDirection", "target", "type"] }, { kind: "component", type: IonCardContent, selector: "ion-card-content", inputs: ["mode"] }, { kind: "component", type: IonIcon, selector: "ion-icon", inputs: ["color", "flipRtl", "icon", "ios", "lazy", "md", "mode", "name", "sanitize", "size", "src"] }, { kind: "component", type: IonSkeletonText, selector: "ion-skeleton-text", inputs: ["animated"] }] }); }
126
141
  }
127
142
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: StatsCardComponent, decorators: [{
128
143
  type: Component,
@@ -168,9 +183,24 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
168
183
  @if (resolvedProps.footer && !resolvedProps.loading) {
169
184
  <div class="stats-footer">{{ resolvedProps.footer }}</div>
170
185
  }
186
+
187
+ @if (resolvedProps.description && !resolvedProps.loading) {
188
+ <div class="stats-description">{{ resolvedProps.description }}</div>
189
+ }
190
+
191
+ @if (resolvedProps.logo && !resolvedProps.loading) {
192
+ <div class="stats-logo">
193
+ <img
194
+ [src]="resolvedProps.logo.src"
195
+ [alt]="resolvedProps.logo.alt"
196
+ [style.max-height.px]="resolvedProps.logo.maxHeight || 32"
197
+ [style.max-width.px]="resolvedProps.logo.maxWidth || 120"
198
+ />
199
+ </div>
200
+ }
171
201
  </ion-card-content>
172
202
  </ion-card>
173
- `, styles: [":host{display:block}.stats-card{margin:0;border-radius:12px;cursor:pointer;transition:transform .2s,box-shadow .2s}.stats-card:hover{transform:translateY(-2px);box-shadow:0 8px 24px #0000001f}.stats-card.background-light{--background: var(--ion-color-light)}.stats-card.background-light .stats-title,.stats-card.background-light .stats-footer{color:var(--ion-color-medium)}.stats-card.background-light .stats-value{color:var(--ion-text-color)}.stats-card.background-light .stats-icon{color:var(--card-color);opacity:.8}.stats-card.background-solid{--background: var(--card-color)}.stats-card.background-solid .stats-title,.stats-card.background-solid .stats-footer,.stats-card.background-solid .stats-value,.stats-card.background-solid .stats-icon{color:#fff}.stats-card.background-solid .stats-title,.stats-card.background-solid .stats-footer{opacity:.9}.stats-card.background-gradient{--background: linear-gradient(135deg, var(--card-color), var(--card-color-shade, var(--card-color)))}.stats-card.background-gradient .stats-title,.stats-card.background-gradient .stats-footer,.stats-card.background-gradient .stats-value,.stats-card.background-gradient .stats-icon{color:#fff}.stats-card.background-gradient .stats-title,.stats-card.background-gradient .stats-footer{opacity:.9}ion-card-content{padding:16px}.stats-header{display:flex;justify-content:space-between;align-items:flex-start;margin-bottom:8px}.stats-title{font-size:13px;font-weight:500;text-transform:uppercase;letter-spacing:.5px}.stats-icon{font-size:24px}.stats-value{font-size:28px;font-weight:700;line-height:1.2;margin-bottom:8px}.stats-value .prefix,.stats-value .suffix{font-size:18px;font-weight:500;opacity:.8}.stats-value .prefix{margin-right:2px}.stats-value .suffix{margin-left:4px}.stats-trend{display:inline-flex;align-items:center;gap:4px;font-size:13px;font-weight:500;padding:4px 8px;border-radius:12px;margin-bottom:8px}.stats-trend ion-icon{font-size:16px}.stats-trend.trend-up{background-color:rgba(var(--ion-color-success-rgb),.15);color:var(--ion-color-success)}.stats-trend.trend-down{background-color:rgba(var(--ion-color-danger-rgb),.15);color:var(--ion-color-danger)}.stats-trend.trend-neutral{background-color:rgba(var(--ion-color-medium-rgb),.15);color:var(--ion-color-medium)}.stats-trend .trend-label{font-weight:400;opacity:.8;margin-left:4px}.stats-footer{font-size:12px;margin-top:4px}\n"] }]
203
+ `, styles: [":host{display:block}.stats-card{margin:0;border-radius:12px;cursor:pointer;transition:transform .2s,box-shadow .2s}.stats-card:hover{transform:translateY(-2px);box-shadow:0 8px 24px #0000001f}.stats-card.background-light{--background: var(--ion-color-light)}.stats-card.background-light .stats-title,.stats-card.background-light .stats-footer{color:var(--ion-color-medium)}.stats-card.background-light .stats-value{color:var(--ion-text-color)}.stats-card.background-light .stats-icon{color:var(--card-color);opacity:.8}.stats-card.background-solid{--background: var(--card-color)}.stats-card.background-solid .stats-title,.stats-card.background-solid .stats-footer,.stats-card.background-solid .stats-value,.stats-card.background-solid .stats-icon{color:#fff}.stats-card.background-solid .stats-title,.stats-card.background-solid .stats-footer{opacity:.9}.stats-card.background-gradient{--background: linear-gradient(135deg, var(--card-color), var(--card-color-shade, var(--card-color)))}.stats-card.background-gradient .stats-title,.stats-card.background-gradient .stats-footer,.stats-card.background-gradient .stats-value,.stats-card.background-gradient .stats-icon{color:#fff}.stats-card.background-gradient .stats-title,.stats-card.background-gradient .stats-footer{opacity:.9}ion-card-content{padding:16px}.stats-header{display:flex;justify-content:space-between;align-items:flex-start;margin-bottom:8px}.stats-title{font-size:13px;font-weight:500;text-transform:uppercase;letter-spacing:.5px}.stats-icon{font-size:24px}.stats-value{font-size:28px;font-weight:700;line-height:1.2;margin-bottom:8px}.stats-value .prefix,.stats-value .suffix{font-size:18px;font-weight:500;opacity:.8}.stats-value .prefix{margin-right:2px}.stats-value .suffix{margin-left:4px}.stats-trend{display:inline-flex;align-items:center;gap:4px;font-size:13px;font-weight:500;padding:4px 8px;border-radius:12px;margin-bottom:8px}.stats-trend ion-icon{font-size:16px}.stats-trend.trend-up{background-color:rgba(var(--ion-color-success-rgb),.15);color:var(--ion-color-success)}.stats-trend.trend-down{background-color:rgba(var(--ion-color-danger-rgb),.15);color:var(--ion-color-danger)}.stats-trend.trend-neutral{background-color:rgba(var(--ion-color-medium-rgb),.15);color:var(--ion-color-medium)}.stats-trend .trend-label{font-weight:400;opacity:.8;margin-left:4px}.stats-footer{font-size:12px;margin-top:4px}.stats-description{margin-top:8px;font-size:14px;line-height:1.4;color:var(--ion-text-color);opacity:.8}.stats-logo{margin-top:16px;padding-top:12px;border-top:1px solid rgba(var(--ion-text-color-rgb),.1)}.stats-logo img{display:block;object-fit:contain;filter:grayscale(100%);opacity:.7;transition:filter .2s,opacity .2s}.stats-logo:hover img{filter:grayscale(0%);opacity:1}.background-solid .stats-description,.background-gradient .stats-description{color:#fff;opacity:.9}.background-solid .stats-logo,.background-gradient .stats-logo{border-top-color:#fff3}.background-solid .stats-logo img,.background-gradient .stats-logo img{filter:grayscale(100%) brightness(10)}.background-solid .stats-logo:hover img,.background-gradient .stats-logo:hover img{filter:brightness(10)}\n"] }]
174
204
  }], propDecorators: { preset: [{
175
205
  type: Input
176
206
  }], props: [{
@@ -178,4 +208,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
178
208
  }], cardClick: [{
179
209
  type: Output
180
210
  }] } });
181
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"stats-card.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/molecules/stats-card/stats-card.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { Component, inject, Input, OnChanges, OnInit, Output, EventEmitter, SimpleChanges } from '@angular/core';\nimport { IonCard, IonCardContent, IonIcon, IonSkeletonText } from '@ionic/angular/standalone';\nimport { CommonModule } from '@angular/common';\nimport { PresetService } from '../../../services/presets';\nimport { StatsCardMetadata } from './types';\nimport { addIcons } from 'ionicons';\nimport { trendingUp, trendingDown, remove, analytics, people, cash, cart, eye, heart, star } from 'ionicons/icons';\n\naddIcons({ trendingUp, trendingDown, remove, analytics, people, cash, cart, eye, heart, star });\n\n@Component({\n  selector: 'val-stats-card',\n  standalone: true,\n  imports: [CommonModule, IonCard, IonCardContent, IonIcon, IonSkeletonText],\n  template: `\n    <ion-card\n      class=\"stats-card\"\n      [class]=\"getBackgroundClass()\"\n      [style.--card-color]=\"getCardColor()\"\n      (click)=\"cardClick.emit()\"\n    >\n      <ion-card-content>\n        <div class=\"stats-header\">\n          <span class=\"stats-title\">{{ resolvedProps.title }}</span>\n          @if (resolvedProps.icon) {\n            <ion-icon [name]=\"resolvedProps.icon\" class=\"stats-icon\"></ion-icon>\n          }\n        </div>\n\n        <div class=\"stats-value\">\n          @if (resolvedProps.loading) {\n            <ion-skeleton-text [animated]=\"true\" style=\"width: 60%; height: 32px;\"></ion-skeleton-text>\n          } @else {\n            @if (resolvedProps.prefix) {\n              <span class=\"prefix\">{{ resolvedProps.prefix }}</span>\n            }\n            <span class=\"value\">{{ resolvedProps.value }}</span>\n            @if (resolvedProps.suffix) {\n              <span class=\"suffix\">{{ resolvedProps.suffix }}</span>\n            }\n          }\n        </div>\n\n        @if (resolvedProps.trend && !resolvedProps.loading) {\n          <div class=\"stats-trend\" [class]=\"getTrendClass()\">\n            <ion-icon [name]=\"getTrendIcon()\"></ion-icon>\n            <span class=\"trend-value\">{{ Math.abs(resolvedProps.trend.value) }}%</span>\n            @if (resolvedProps.trend.label) {\n              <span class=\"trend-label\">{{ resolvedProps.trend.label }}</span>\n            }\n          </div>\n        }\n\n        @if (resolvedProps.footer && !resolvedProps.loading) {\n          <div class=\"stats-footer\">{{ resolvedProps.footer }}</div>\n        }\n      </ion-card-content>\n    </ion-card>\n  `,\n  styleUrls: ['./stats-card.component.scss'],\n})\n/**\n * val-stats-card\n *\n * A card component for displaying statistics and KPIs.\n * Supports presets for reusable configurations.\n *\n * @example With preset:\n * <val-stats-card preset=\"kpi\" [props]=\"{ title: 'Users', value: 1000 }\"></val-stats-card>\n *\n * @example Basic usage\n * <val-stats-card [props]=\"{\n *   title: 'Total Users',\n *   value: 12500,\n *   icon: 'people',\n *   color: 'primary'\n * }\"></val-stats-card>\n *\n * @input preset: string - Name of preset to apply\n * @input props: StatsCardMetadata - Configuration for the stats card\n * @output cardClick: void - Emits when card is clicked\n */\nexport class StatsCardComponent implements OnInit, OnChanges {\n  private presets = inject(PresetService);\n\n  @Input() preset?: string;\n  @Input() props: Partial<StatsCardMetadata> = {};\n\n  resolvedProps: StatsCardMetadata = {} as StatsCardMetadata;\n\n  @Output() cardClick = new EventEmitter<void>();\n\n  Math = Math;\n\n  ngOnInit() {\n    this.resolveProps();\n  }\n\n  ngOnChanges(changes: SimpleChanges) {\n    if (changes['preset'] || changes['props']) {\n      this.resolveProps();\n    }\n  }\n\n  private resolveProps(): void {\n    const presetProps = this.preset\n      ? (this.presets.get('statsCard', this.preset) as Partial<StatsCardMetadata>)\n      : {};\n\n    this.resolvedProps = {\n      ...presetProps,\n      ...this.props,\n    } as StatsCardMetadata;\n  }\n\n  getCardColor(): string {\n    const color = this.resolvedProps.color || 'primary';\n    return `var(--ion-color-${color})`;\n  }\n\n  getBackgroundClass(): string {\n    const bg = this.resolvedProps.background || 'light';\n    return `background-${bg}`;\n  }\n\n  getTrendClass(): string {\n    const direction = this.resolvedProps.trend?.direction || 'neutral';\n    const invert = this.resolvedProps.trend?.invertColors;\n\n    if (direction === 'neutral') return 'trend-neutral';\n    if (direction === 'up') return invert ? 'trend-down' : 'trend-up';\n    return invert ? 'trend-up' : 'trend-down';\n  }\n\n  getTrendIcon(): string {\n    const direction = this.resolvedProps.trend?.direction || 'neutral';\n    if (direction === 'up') return 'trending-up';\n    if (direction === 'down') return 'trending-down';\n    return 'remove';\n  }\n}\n"]}
211
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"stats-card.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/molecules/stats-card/stats-card.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { Component, inject, Input, OnChanges, OnInit, Output, EventEmitter, SimpleChanges } from '@angular/core';\nimport { IonCard, IonCardContent, IonIcon, IonSkeletonText } from '@ionic/angular/standalone';\nimport { CommonModule } from '@angular/common';\nimport { PresetService } from '../../../services/presets';\nimport { StatsCardMetadata } from './types';\nimport { addIcons } from 'ionicons';\nimport { trendingUp, trendingDown, remove, analytics, people, cash, cart, eye, heart, star } from 'ionicons/icons';\n\naddIcons({ trendingUp, trendingDown, remove, analytics, people, cash, cart, eye, heart, star });\n\n@Component({\n  selector: 'val-stats-card',\n  standalone: true,\n  imports: [CommonModule, IonCard, IonCardContent, IonIcon, IonSkeletonText],\n  template: `\n    <ion-card\n      class=\"stats-card\"\n      [class]=\"getBackgroundClass()\"\n      [style.--card-color]=\"getCardColor()\"\n      (click)=\"cardClick.emit()\"\n    >\n      <ion-card-content>\n        <div class=\"stats-header\">\n          <span class=\"stats-title\">{{ resolvedProps.title }}</span>\n          @if (resolvedProps.icon) {\n            <ion-icon [name]=\"resolvedProps.icon\" class=\"stats-icon\"></ion-icon>\n          }\n        </div>\n\n        <div class=\"stats-value\">\n          @if (resolvedProps.loading) {\n            <ion-skeleton-text [animated]=\"true\" style=\"width: 60%; height: 32px;\"></ion-skeleton-text>\n          } @else {\n            @if (resolvedProps.prefix) {\n              <span class=\"prefix\">{{ resolvedProps.prefix }}</span>\n            }\n            <span class=\"value\">{{ resolvedProps.value }}</span>\n            @if (resolvedProps.suffix) {\n              <span class=\"suffix\">{{ resolvedProps.suffix }}</span>\n            }\n          }\n        </div>\n\n        @if (resolvedProps.trend && !resolvedProps.loading) {\n          <div class=\"stats-trend\" [class]=\"getTrendClass()\">\n            <ion-icon [name]=\"getTrendIcon()\"></ion-icon>\n            <span class=\"trend-value\">{{ Math.abs(resolvedProps.trend.value) }}%</span>\n            @if (resolvedProps.trend.label) {\n              <span class=\"trend-label\">{{ resolvedProps.trend.label }}</span>\n            }\n          </div>\n        }\n\n        @if (resolvedProps.footer && !resolvedProps.loading) {\n          <div class=\"stats-footer\">{{ resolvedProps.footer }}</div>\n        }\n\n        @if (resolvedProps.description && !resolvedProps.loading) {\n          <div class=\"stats-description\">{{ resolvedProps.description }}</div>\n        }\n\n        @if (resolvedProps.logo && !resolvedProps.loading) {\n          <div class=\"stats-logo\">\n            <img\n              [src]=\"resolvedProps.logo.src\"\n              [alt]=\"resolvedProps.logo.alt\"\n              [style.max-height.px]=\"resolvedProps.logo.maxHeight || 32\"\n              [style.max-width.px]=\"resolvedProps.logo.maxWidth || 120\"\n            />\n          </div>\n        }\n      </ion-card-content>\n    </ion-card>\n  `,\n  styleUrls: ['./stats-card.component.scss'],\n})\n/**\n * val-stats-card\n *\n * A card component for displaying statistics and KPIs.\n * Supports presets for reusable configurations.\n *\n * @example With preset:\n * <val-stats-card preset=\"kpi\" [props]=\"{ title: 'Users', value: 1000 }\"></val-stats-card>\n *\n * @example Basic usage\n * <val-stats-card [props]=\"{\n *   title: 'Total Users',\n *   value: 12500,\n *   icon: 'people',\n *   color: 'primary'\n * }\"></val-stats-card>\n *\n * @input preset: string - Name of preset to apply\n * @input props: StatsCardMetadata - Configuration for the stats card\n * @output cardClick: void - Emits when card is clicked\n */\nexport class StatsCardComponent implements OnInit, OnChanges {\n  private presets = inject(PresetService);\n\n  @Input() preset?: string;\n  @Input() props: Partial<StatsCardMetadata> = {};\n\n  resolvedProps: StatsCardMetadata = {} as StatsCardMetadata;\n\n  @Output() cardClick = new EventEmitter<void>();\n\n  Math = Math;\n\n  ngOnInit() {\n    this.resolveProps();\n  }\n\n  ngOnChanges(changes: SimpleChanges) {\n    if (changes['preset'] || changes['props']) {\n      this.resolveProps();\n    }\n  }\n\n  private resolveProps(): void {\n    const presetProps = this.preset\n      ? (this.presets.get('statsCard', this.preset) as Partial<StatsCardMetadata>)\n      : {};\n\n    this.resolvedProps = {\n      ...presetProps,\n      ...this.props,\n    } as StatsCardMetadata;\n  }\n\n  getCardColor(): string {\n    const color = this.resolvedProps.color || 'primary';\n    return `var(--ion-color-${color})`;\n  }\n\n  getBackgroundClass(): string {\n    const bg = this.resolvedProps.background || 'light';\n    return `background-${bg}`;\n  }\n\n  getTrendClass(): string {\n    const direction = this.resolvedProps.trend?.direction || 'neutral';\n    const invert = this.resolvedProps.trend?.invertColors;\n\n    if (direction === 'neutral') return 'trend-neutral';\n    if (direction === 'up') return invert ? 'trend-down' : 'trend-up';\n    return invert ? 'trend-up' : 'trend-down';\n  }\n\n  getTrendIcon(): string {\n    const direction = this.resolvedProps.trend?.direction || 'neutral';\n    if (direction === 'up') return 'trending-up';\n    if (direction === 'down') return 'trending-down';\n    return 'remove';\n  }\n}\n"]}
@@ -1,2 +1,2 @@
1
1
  export {};
2
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL2NvbXBvbmVudHMvbW9sZWN1bGVzL3N0YXRzLWNhcmQvdHlwZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENvbG9yIH0gZnJvbSAnQGlvbmljL2NvcmUnO1xuXG4vKipcbiAqIFRyZW5kIGNvbmZpZ3VyYXRpb24gZm9yIHN0YXRzIGNhcmQuXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgU3RhdHNUcmVuZCB7XG4gIC8qKiBUcmVuZCB2YWx1ZSAocGVyY2VudGFnZSBvciBhYnNvbHV0ZSkgKi9cbiAgdmFsdWU6IG51bWJlcjtcbiAgLyoqIFRyZW5kIGRpcmVjdGlvbiAqL1xuICBkaXJlY3Rpb246ICd1cCcgfCAnZG93bicgfCAnbmV1dHJhbCc7XG4gIC8qKiBUcmVuZCBsYWJlbCAoZS5nLiwgXCJ2cyBsYXN0IG1vbnRoXCIpICovXG4gIGxhYmVsPzogc3RyaW5nO1xuICAvKiogSW52ZXJ0IGNvbG9ycyAodXA9YmFkLCBkb3duPWdvb2QpICovXG4gIGludmVydENvbG9ycz86IGJvb2xlYW47XG59XG5cbi8qKlxuICogTWV0YWRhdGEgZm9yIHRoZSBzdGF0cyBjYXJkIGNvbXBvbmVudC5cbiAqL1xuZXhwb3J0IGludGVyZmFjZSBTdGF0c0NhcmRNZXRhZGF0YSB7XG4gIC8qKiBTdGF0IHRpdGxlL2xhYmVsICovXG4gIHRpdGxlOiBzdHJpbmc7XG4gIC8qKiBNYWluIHN0YXQgdmFsdWUgKi9cbiAgdmFsdWU6IHN0cmluZyB8IG51bWJlcjtcbiAgLyoqIFZhbHVlIHByZWZpeCAoZS5nLiwgXCIkXCIsIFwi4oKsXCIpICovXG4gIHByZWZpeD86IHN0cmluZztcbiAgLyoqIFZhbHVlIHN1ZmZpeCAoZS5nLiwgXCIlXCIsIFwidXNlcnNcIikgKi9cbiAgc3VmZml4Pzogc3RyaW5nO1xuICAvKiogSWNvbiBuYW1lICovXG4gIGljb24/OiBzdHJpbmc7XG4gIC8qKiBDYXJkIGNvbG9yICovXG4gIGNvbG9yPzogQ29sb3I7XG4gIC8qKiBCYWNrZ3JvdW5kIHN0eWxlICovXG4gIGJhY2tncm91bmQ/OiAnc29saWQnIHwgJ2dyYWRpZW50JyB8ICdsaWdodCc7XG4gIC8qKiBUcmVuZCBpbmRpY2F0b3IgKi9cbiAgdHJlbmQ/OiBTdGF0c1RyZW5kO1xuICAvKiogRm9vdGVyIHRleHQgKi9cbiAgZm9vdGVyPzogc3RyaW5nO1xuICAvKiogTG9hZGluZyBzdGF0ZSAqL1xuICBsb2FkaW5nPzogYm9vbGVhbjtcbiAgLyoqIFVuaXF1ZSB0b2tlbiBpZGVudGlmaWVyICovXG4gIHRva2VuPzogc3RyaW5nO1xufVxuIl19
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL2NvbXBvbmVudHMvbW9sZWN1bGVzL3N0YXRzLWNhcmQvdHlwZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENvbG9yIH0gZnJvbSAnQGlvbmljL2NvcmUnO1xuXG4vKipcbiAqIExvZ28gY29uZmlndXJhdGlvbiBmb3Igc3RhdHMgY2FyZC5cbiAqL1xuZXhwb3J0IGludGVyZmFjZSBTdGF0c0NhcmRMb2dvIHtcbiAgLyoqIExvZ28gaW1hZ2UgVVJMICovXG4gIHNyYzogc3RyaW5nO1xuICAvKiogQWx0IHRleHQgZm9yIGFjY2Vzc2liaWxpdHkgKi9cbiAgYWx0OiBzdHJpbmc7XG4gIC8qKiBNYXggaGVpZ2h0IGluIHBpeGVscyAoZGVmYXVsdDogMzIpICovXG4gIG1heEhlaWdodD86IG51bWJlcjtcbiAgLyoqIE1heCB3aWR0aCBpbiBwaXhlbHMgKGRlZmF1bHQ6IDEyMCkgKi9cbiAgbWF4V2lkdGg/OiBudW1iZXI7XG59XG5cbi8qKlxuICogVHJlbmQgY29uZmlndXJhdGlvbiBmb3Igc3RhdHMgY2FyZC5cbiAqL1xuZXhwb3J0IGludGVyZmFjZSBTdGF0c1RyZW5kIHtcbiAgLyoqIFRyZW5kIHZhbHVlIChwZXJjZW50YWdlIG9yIGFic29sdXRlKSAqL1xuICB2YWx1ZTogbnVtYmVyO1xuICAvKiogVHJlbmQgZGlyZWN0aW9uICovXG4gIGRpcmVjdGlvbjogJ3VwJyB8ICdkb3duJyB8ICduZXV0cmFsJztcbiAgLyoqIFRyZW5kIGxhYmVsIChlLmcuLCBcInZzIGxhc3QgbW9udGhcIikgKi9cbiAgbGFiZWw/OiBzdHJpbmc7XG4gIC8qKiBJbnZlcnQgY29sb3JzICh1cD1iYWQsIGRvd249Z29vZCkgKi9cbiAgaW52ZXJ0Q29sb3JzPzogYm9vbGVhbjtcbn1cblxuLyoqXG4gKiBNZXRhZGF0YSBmb3IgdGhlIHN0YXRzIGNhcmQgY29tcG9uZW50LlxuICovXG5leHBvcnQgaW50ZXJmYWNlIFN0YXRzQ2FyZE1ldGFkYXRhIHtcbiAgLyoqIFN0YXQgdGl0bGUvbGFiZWwgKi9cbiAgdGl0bGU6IHN0cmluZztcbiAgLyoqIE1haW4gc3RhdCB2YWx1ZSAqL1xuICB2YWx1ZTogc3RyaW5nIHwgbnVtYmVyO1xuICAvKiogVmFsdWUgcHJlZml4IChlLmcuLCBcIiRcIiwgXCLigqxcIikgKi9cbiAgcHJlZml4Pzogc3RyaW5nO1xuICAvKiogVmFsdWUgc3VmZml4IChlLmcuLCBcIiVcIiwgXCJ1c2Vyc1wiKSAqL1xuICBzdWZmaXg/OiBzdHJpbmc7XG4gIC8qKiBJY29uIG5hbWUgKi9cbiAgaWNvbj86IHN0cmluZztcbiAgLyoqIENhcmQgY29sb3IgKi9cbiAgY29sb3I/OiBDb2xvcjtcbiAgLyoqIEJhY2tncm91bmQgc3R5bGUgKi9cbiAgYmFja2dyb3VuZD86ICdzb2xpZCcgfCAnZ3JhZGllbnQnIHwgJ2xpZ2h0JztcbiAgLyoqIFRyZW5kIGluZGljYXRvciAqL1xuICB0cmVuZD86IFN0YXRzVHJlbmQ7XG4gIC8qKiBGb290ZXIgdGV4dCAqL1xuICBmb290ZXI/OiBzdHJpbmc7XG4gIC8qKiBEZXNjcmlwdGlvbiB0ZXh0IChmb3IgR2l0TGFiLXN0eWxlIGxheW91dCkgKi9cbiAgZGVzY3JpcHRpb24/OiBzdHJpbmc7XG4gIC8qKiBDb21wYW55L2JyYW5kIGxvZ28gZGlzcGxheWVkIGF0IGJvdHRvbSAqL1xuICBsb2dvPzogU3RhdHNDYXJkTG9nbztcbiAgLyoqIExvYWRpbmcgc3RhdGUgKi9cbiAgbG9hZGluZz86IGJvb2xlYW47XG4gIC8qKiBVbmlxdWUgdG9rZW4gaWRlbnRpZmllciAqL1xuICB0b2tlbj86IHN0cmluZztcbn1cbiJdfQ==
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL2NvbXBvbmVudHMvbW9sZWN1bGVzL3VzZXJuYW1lLWlucHV0L3R5cGVzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBGb3JtQ29udHJvbCB9IGZyb20gJ0Bhbmd1bGFyL2Zvcm1zJztcbmltcG9ydCB7IE9ic2VydmFibGUgfSBmcm9tICdyeGpzJztcbmltcG9ydCB7IENvbXBvbmVudFN0YXRlIH0gZnJvbSAnLi4vLi4vdHlwZXMnO1xuXG4vKipcbiAqIE1ldGFkYXRhIGZvciBVc2VybmFtZUlucHV0Q29tcG9uZW50LlxuICovXG5leHBvcnQgaW50ZXJmYWNlIFVzZXJuYW1lSW5wdXRNZXRhZGF0YSB7XG4gIC8qKiBGb3JtQ29udHJvbCBwYXJhIGVsIHZhbG9yIGRlbCB1c2VybmFtZSAqL1xuICBjb250cm9sOiBGb3JtQ29udHJvbDxzdHJpbmc+O1xuICAvKiogTGFiZWwgZGVsIGNhbXBvICovXG4gIGxhYmVsPzogc3RyaW5nO1xuICAvKiogUGxhY2Vob2xkZXIgZGVsIGlucHV0ICovXG4gIHBsYWNlaG9sZGVyPzogc3RyaW5nO1xuICAvKiogUHJlZmlqbyB2aXN1YWwgKGRlZmF1bHQ6ICdAJykgKi9cbiAgcHJlZml4Pzogc3RyaW5nO1xuICAvKiogTG9uZ2l0dWQgbcOtbmltYSBwZXJtaXRpZGEgKGRlZmF1bHQ6IDMpICovXG4gIG1pbkxlbmd0aD86IG51bWJlcjtcbiAgLyoqIExvbmdpdHVkIG3DoXhpbWEgcGVybWl0aWRhIChkZWZhdWx0OiAzMCkgKi9cbiAgbWF4TGVuZ3RoPzogbnVtYmVyO1xuICAvKiogRnVuY2nDs24gcGFyYSB2ZXJpZmljYXIgZGlzcG9uaWJpbGlkYWQgZGVsIHVzZXJuYW1lICovXG4gIGNoZWNrQXZhaWxhYmlsaXR5PzogKGhhbmRsZTogc3RyaW5nKSA9PiBPYnNlcnZhYmxlPGJvb2xlYW4+O1xuICAvKiogTWVuc2FqZXMgZGUgZXJyb3IgcGVyc29uYWxpemFkb3MgKi9cbiAgZXJyb3JzPzogUmVjb3JkPHN0cmluZywgc3RyaW5nPjtcbiAgLyoqIEVzdGFkbyBkZWwgY29tcG9uZW50ZSAqL1xuICBzdGF0ZT86IENvbXBvbmVudFN0YXRlO1xuICAvKiogTW9zdHJhciBpbmRpY2Fkb3IgZGUgZGlzcG9uaWJpbGlkYWQgKGRlZmF1bHQ6IHRydWUpICovXG4gIHNob3dBdmFpbGFiaWxpdHk/OiBib29sZWFuO1xuICAvKiogRGVib3VuY2UgZW4gbXMgcGFyYSBjaGVjayBkZSBkaXNwb25pYmlsaWRhZCAoZGVmYXVsdDogNTAwKSAqL1xuICBkZWJvdW5jZVRpbWU/OiBudW1iZXI7XG59XG5cbi8qKlxuICogRXN0YWRvIGRlIGRpc3BvbmliaWxpZGFkIGRlbCB1c2VybmFtZS5cbiAqL1xuZXhwb3J0IHR5cGUgVXNlcm5hbWVBdmFpbGFiaWxpdHlTdGF0dXMgPSAnaWRsZScgfCAnY2hlY2tpbmcnIHwgJ2F2YWlsYWJsZScgfCAndGFrZW4nIHwgJ2ludmFsaWQnO1xuIl19
@@ -0,0 +1,260 @@
1
+ import { Component, Input, signal, computed, } from '@angular/core';
2
+ import { CommonModule } from '@angular/common';
3
+ import { ReactiveFormsModule } from '@angular/forms';
4
+ import { IonInput, IonSpinner, IonIcon, IonText } from '@ionic/angular/standalone';
5
+ import { Subject, debounceTime, distinctUntilChanged, takeUntil, switchMap, of, catchError } from 'rxjs';
6
+ import { addIcons } from 'ionicons';
7
+ import { checkmarkCircle, closeCircle, alertCircle } from 'ionicons/icons';
8
+ import * as i0 from "@angular/core";
9
+ import * as i1 from "@angular/forms";
10
+ addIcons({ checkmarkCircle, closeCircle, alertCircle });
11
+ /**
12
+ * Username Input Component
13
+ *
14
+ * Input especializado para usernames/handles con:
15
+ * - Prefijo '@' visual
16
+ * - Validación de formato (alfanuméricos y _)
17
+ * - Normalización automática (lowercase, sin espacios)
18
+ * - Verificación de disponibilidad con debounce
19
+ * - Estados visuales: available, taken, checking
20
+ *
21
+ * @example
22
+ * <val-username-input
23
+ * [props]="{
24
+ * control: usernameControl,
25
+ * label: 'Nombre de usuario',
26
+ * placeholder: 'tu_username',
27
+ * checkAvailability: checkFn
28
+ * }"
29
+ * />
30
+ */
31
+ export class UsernameInputComponent {
32
+ constructor() {
33
+ this.destroy$ = new Subject();
34
+ this.checkAvailability$ = new Subject();
35
+ // Signals
36
+ this.isFocused = signal(false);
37
+ this.availabilityStatus = signal('idle');
38
+ // Computed
39
+ this.hasError = computed(() => {
40
+ const control = this.props?.control;
41
+ return control?.touched && control?.invalid;
42
+ });
43
+ this.showStatusMessage = computed(() => {
44
+ const status = this.availabilityStatus();
45
+ return status === 'available' || status === 'taken';
46
+ });
47
+ this.statusColor = computed(() => {
48
+ return this.availabilityStatus() === 'available' ? 'success' : 'danger';
49
+ });
50
+ this.statusMessage = computed(() => {
51
+ const status = this.availabilityStatus();
52
+ if (status === 'available')
53
+ return 'Username disponible';
54
+ if (status === 'taken')
55
+ return 'Username ya está en uso';
56
+ return '';
57
+ });
58
+ this.errorMessage = computed(() => {
59
+ const control = this.props?.control;
60
+ if (!control?.errors)
61
+ return '';
62
+ const errors = this.props?.errors || {};
63
+ if (control.errors['required']) {
64
+ return errors['required'] || 'El username es requerido';
65
+ }
66
+ if (control.errors['minlength']) {
67
+ const min = this.props?.minLength || 3;
68
+ return errors['minlength'] || `Mínimo ${min} caracteres`;
69
+ }
70
+ if (control.errors['maxlength']) {
71
+ const max = this.props?.maxLength || 30;
72
+ return errors['maxlength'] || `Máximo ${max} caracteres`;
73
+ }
74
+ if (control.errors['pattern']) {
75
+ return errors['pattern'] || 'Solo letras, números y guión bajo (_)';
76
+ }
77
+ return '';
78
+ });
79
+ }
80
+ ngOnInit() {
81
+ this.setupAvailabilityCheck();
82
+ }
83
+ ngOnDestroy() {
84
+ this.destroy$.next();
85
+ this.destroy$.complete();
86
+ }
87
+ onFocus() {
88
+ this.isFocused.set(true);
89
+ }
90
+ onBlur() {
91
+ this.isFocused.set(false);
92
+ this.props.control.markAsTouched();
93
+ }
94
+ onInput(event) {
95
+ const input = event.detail.value || '';
96
+ // Normalize: lowercase, remove spaces, only allow valid chars
97
+ const normalized = this.normalizeUsername(input);
98
+ if (normalized !== input) {
99
+ this.props.control.setValue(normalized, { emitEvent: false });
100
+ }
101
+ // Trigger availability check
102
+ if (this.props.checkAvailability && this.isValidFormat(normalized)) {
103
+ this.checkAvailability$.next(normalized);
104
+ }
105
+ else if (!this.isValidFormat(normalized) && normalized.length > 0) {
106
+ this.availabilityStatus.set('invalid');
107
+ }
108
+ else {
109
+ this.availabilityStatus.set('idle');
110
+ }
111
+ }
112
+ setupAvailabilityCheck() {
113
+ if (!this.props?.checkAvailability)
114
+ return;
115
+ this.checkAvailability$
116
+ .pipe(debounceTime(this.props.debounceTime || 500), distinctUntilChanged(), switchMap(username => {
117
+ if (!username || username.length < (this.props.minLength || 3)) {
118
+ return of(null);
119
+ }
120
+ this.availabilityStatus.set('checking');
121
+ return this.props.checkAvailability(username).pipe(catchError(() => of(null)));
122
+ }), takeUntil(this.destroy$))
123
+ .subscribe(available => {
124
+ if (available === null) {
125
+ this.availabilityStatus.set('idle');
126
+ }
127
+ else if (available) {
128
+ this.availabilityStatus.set('available');
129
+ }
130
+ else {
131
+ this.availabilityStatus.set('taken');
132
+ }
133
+ });
134
+ }
135
+ normalizeUsername(value) {
136
+ return value
137
+ .toLowerCase()
138
+ .replace(/\s/g, '')
139
+ .replace(/[^a-z0-9_]/g, '');
140
+ }
141
+ isValidFormat(value) {
142
+ if (!value)
143
+ return false;
144
+ const minLen = this.props?.minLength || 3;
145
+ const maxLen = this.props?.maxLength || 30;
146
+ const pattern = /^[a-z0-9_]+$/;
147
+ return value.length >= minLen && value.length <= maxLen && pattern.test(value);
148
+ }
149
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UsernameInputComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
150
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: UsernameInputComponent, isStandalone: true, selector: "val-username-input", inputs: { props: "props" }, ngImport: i0, template: `
151
+ <div class="username-input-container">
152
+ @if (props.label) {
153
+ <label class="username-label">{{ props.label }}</label>
154
+ }
155
+
156
+ <div class="username-input-wrapper" [class.focused]="isFocused()" [class.error]="hasError()">
157
+ <span class="username-prefix">{{ props.prefix || '@' }}</span>
158
+ <ion-input
159
+ [formControl]="props.control"
160
+ type="text"
161
+ [placeholder]="props.placeholder || 'username'"
162
+ [maxlength]="props.maxLength || 30"
163
+ (ionFocus)="onFocus()"
164
+ (ionBlur)="onBlur()"
165
+ (ionInput)="onInput($event)"
166
+ class="username-field"
167
+ />
168
+
169
+ @if (props.showAvailability !== false) {
170
+ <div class="availability-indicator">
171
+ @switch (availabilityStatus()) {
172
+ @case ('checking') {
173
+ <ion-spinner name="crescent" class="checking-spinner" />
174
+ }
175
+ @case ('available') {
176
+ <ion-icon name="checkmark-circle" class="status-icon available" />
177
+ }
178
+ @case ('taken') {
179
+ <ion-icon name="close-circle" class="status-icon taken" />
180
+ }
181
+ @case ('invalid') {
182
+ <ion-icon name="alert-circle" class="status-icon invalid" />
183
+ }
184
+ }
185
+ </div>
186
+ }
187
+ </div>
188
+
189
+ @if (showStatusMessage()) {
190
+ <ion-text [color]="statusColor()" class="status-message">
191
+ <small>{{ statusMessage() }}</small>
192
+ </ion-text>
193
+ }
194
+
195
+ @if (hasError() && errorMessage()) {
196
+ <ion-text color="danger" class="error-message">
197
+ <small>{{ errorMessage() }}</small>
198
+ </ion-text>
199
+ }
200
+ </div>
201
+ `, isInline: true, styles: [".username-input-container{margin-bottom:1rem}.username-label{display:block;font-size:.875rem;font-weight:500;color:var(--ion-color-dark);margin-bottom:.5rem}.username-input-wrapper{display:flex;align-items:center;background:var(--ion-background-color, #fff);border:1px solid var(--ion-border-color, #e0e0e0);border-radius:8px;padding:0 .75rem;transition:border-color .2s,box-shadow .2s}.username-input-wrapper.focused{border-color:var(--ion-color-primary);box-shadow:0 0 0 2px var(--ion-color-primary-tint)}.username-input-wrapper.error{border-color:var(--ion-color-danger)}.username-prefix{font-size:1rem;font-weight:500;color:var(--ion-color-medium);-webkit-user-select:none;user-select:none}.username-field{flex:1;--padding-start: .25rem;--padding-end: 0;--background: transparent;font-size:1rem}.username-field::part(native){padding-left:.25rem}.availability-indicator{display:flex;align-items:center;justify-content:center;width:24px;height:24px;margin-left:.5rem}.checking-spinner{width:18px;height:18px;--color: var(--ion-color-medium)}.status-icon{font-size:1.25rem}.status-icon.available{color:var(--ion-color-success)}.status-icon.taken{color:var(--ion-color-danger)}.status-icon.invalid{color:var(--ion-color-warning)}.status-message,.error-message{display:block;margin-top:.25rem;padding-left:.25rem}:host-context(body.dark){.username-input-wrapper{background:var(--ion-color-step-50);border-color:var(--ion-color-step-150)}.username-label{color:var(--ion-color-light)}}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i1.MaxLengthValidator, selector: "[maxlength][formControlName],[maxlength][formControl],[maxlength][ngModel]", inputs: ["maxlength"] }, { kind: "directive", type: i1.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "component", type: IonInput, selector: "ion-input", inputs: ["accept", "autocapitalize", "autocomplete", "autocorrect", "autofocus", "clearInput", "clearOnEdit", "color", "counter", "counterFormatter", "debounce", "disabled", "enterkeyhint", "errorText", "fill", "helperText", "inputmode", "label", "labelPlacement", "max", "maxlength", "min", "minlength", "mode", "multiple", "name", "pattern", "placeholder", "readonly", "required", "shape", "size", "spellcheck", "step", "type", "value"] }, { kind: "component", type: IonSpinner, selector: "ion-spinner", inputs: ["color", "duration", "name", "paused"] }, { kind: "component", type: IonIcon, selector: "ion-icon", inputs: ["color", "flipRtl", "icon", "ios", "lazy", "md", "mode", "name", "sanitize", "size", "src"] }, { kind: "component", type: IonText, selector: "ion-text", inputs: ["color", "mode"] }] }); }
202
+ }
203
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: UsernameInputComponent, decorators: [{
204
+ type: Component,
205
+ args: [{ selector: 'val-username-input', standalone: true, imports: [CommonModule, ReactiveFormsModule, IonInput, IonSpinner, IonIcon, IonText], template: `
206
+ <div class="username-input-container">
207
+ @if (props.label) {
208
+ <label class="username-label">{{ props.label }}</label>
209
+ }
210
+
211
+ <div class="username-input-wrapper" [class.focused]="isFocused()" [class.error]="hasError()">
212
+ <span class="username-prefix">{{ props.prefix || '@' }}</span>
213
+ <ion-input
214
+ [formControl]="props.control"
215
+ type="text"
216
+ [placeholder]="props.placeholder || 'username'"
217
+ [maxlength]="props.maxLength || 30"
218
+ (ionFocus)="onFocus()"
219
+ (ionBlur)="onBlur()"
220
+ (ionInput)="onInput($event)"
221
+ class="username-field"
222
+ />
223
+
224
+ @if (props.showAvailability !== false) {
225
+ <div class="availability-indicator">
226
+ @switch (availabilityStatus()) {
227
+ @case ('checking') {
228
+ <ion-spinner name="crescent" class="checking-spinner" />
229
+ }
230
+ @case ('available') {
231
+ <ion-icon name="checkmark-circle" class="status-icon available" />
232
+ }
233
+ @case ('taken') {
234
+ <ion-icon name="close-circle" class="status-icon taken" />
235
+ }
236
+ @case ('invalid') {
237
+ <ion-icon name="alert-circle" class="status-icon invalid" />
238
+ }
239
+ }
240
+ </div>
241
+ }
242
+ </div>
243
+
244
+ @if (showStatusMessage()) {
245
+ <ion-text [color]="statusColor()" class="status-message">
246
+ <small>{{ statusMessage() }}</small>
247
+ </ion-text>
248
+ }
249
+
250
+ @if (hasError() && errorMessage()) {
251
+ <ion-text color="danger" class="error-message">
252
+ <small>{{ errorMessage() }}</small>
253
+ </ion-text>
254
+ }
255
+ </div>
256
+ `, styles: [".username-input-container{margin-bottom:1rem}.username-label{display:block;font-size:.875rem;font-weight:500;color:var(--ion-color-dark);margin-bottom:.5rem}.username-input-wrapper{display:flex;align-items:center;background:var(--ion-background-color, #fff);border:1px solid var(--ion-border-color, #e0e0e0);border-radius:8px;padding:0 .75rem;transition:border-color .2s,box-shadow .2s}.username-input-wrapper.focused{border-color:var(--ion-color-primary);box-shadow:0 0 0 2px var(--ion-color-primary-tint)}.username-input-wrapper.error{border-color:var(--ion-color-danger)}.username-prefix{font-size:1rem;font-weight:500;color:var(--ion-color-medium);-webkit-user-select:none;user-select:none}.username-field{flex:1;--padding-start: .25rem;--padding-end: 0;--background: transparent;font-size:1rem}.username-field::part(native){padding-left:.25rem}.availability-indicator{display:flex;align-items:center;justify-content:center;width:24px;height:24px;margin-left:.5rem}.checking-spinner{width:18px;height:18px;--color: var(--ion-color-medium)}.status-icon{font-size:1.25rem}.status-icon.available{color:var(--ion-color-success)}.status-icon.taken{color:var(--ion-color-danger)}.status-icon.invalid{color:var(--ion-color-warning)}.status-message,.error-message{display:block;margin-top:.25rem;padding-left:.25rem}:host-context(body.dark){.username-input-wrapper{background:var(--ion-color-step-50);border-color:var(--ion-color-step-150)}.username-label{color:var(--ion-color-light)}}\n"] }]
257
+ }], propDecorators: { props: [{
258
+ type: Input
259
+ }] } });
260
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"username-input.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/molecules/username-input/username-input.component.ts"],"names":[],"mappings":"AAAA,OAAO,EACL,SAAS,EACT,KAAK,EAGL,MAAM,EACN,QAAQ,GAET,MAAM,eAAe,CAAC;AACvB,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;AAC/C,OAAO,EAAE,mBAAmB,EAAE,MAAM,gBAAgB,CAAC;AACrD,OAAO,EAAE,QAAQ,EAAE,UAAU,EAAE,OAAO,EAAE,OAAO,EAAE,MAAM,2BAA2B,CAAC;AACnF,OAAO,EAAE,OAAO,EAAE,YAAY,EAAE,oBAAoB,EAAE,SAAS,EAAE,SAAS,EAAE,EAAE,EAAE,UAAU,EAAE,MAAM,MAAM,CAAC;AACzG,OAAO,EAAE,QAAQ,EAAE,MAAM,UAAU,CAAC;AACpC,OAAO,EAAE,eAAe,EAAE,WAAW,EAAE,WAAW,EAAE,MAAM,gBAAgB,CAAC;;;AAG3E,QAAQ,CAAC,EAAE,eAAe,EAAE,WAAW,EAAE,WAAW,EAAE,CAAC,CAAC;AAExD;;;;;;;;;;;;;;;;;;;GAmBG;AA+JH,MAAM,OAAO,sBAAsB;IA9JnC;QAiKU,aAAQ,GAAG,IAAI,OAAO,EAAQ,CAAC;QAC/B,uBAAkB,GAAG,IAAI,OAAO,EAAU,CAAC;QAEnD,UAAU;QACV,cAAS,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC;QAC1B,uBAAkB,GAAG,MAAM,CAA6B,MAAM,CAAC,CAAC;QAEhE,WAAW;QACX,aAAQ,GAAG,QAAQ,CAAC,GAAG,EAAE;YACvB,MAAM,OAAO,GAAG,IAAI,CAAC,KAAK,EAAE,OAAO,CAAC;YACpC,OAAO,OAAO,EAAE,OAAO,IAAI,OAAO,EAAE,OAAO,CAAC;QAC9C,CAAC,CAAC,CAAC;QAEH,sBAAiB,GAAG,QAAQ,CAAC,GAAG,EAAE;YAChC,MAAM,MAAM,GAAG,IAAI,CAAC,kBAAkB,EAAE,CAAC;YACzC,OAAO,MAAM,KAAK,WAAW,IAAI,MAAM,KAAK,OAAO,CAAC;QACtD,CAAC,CAAC,CAAC;QAEH,gBAAW,GAAG,QAAQ,CAAC,GAAG,EAAE;YAC1B,OAAO,IAAI,CAAC,kBAAkB,EAAE,KAAK,WAAW,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,QAAQ,CAAC;QAC1E,CAAC,CAAC,CAAC;QAEH,kBAAa,GAAG,QAAQ,CAAC,GAAG,EAAE;YAC5B,MAAM,MAAM,GAAG,IAAI,CAAC,kBAAkB,EAAE,CAAC;YACzC,IAAI,MAAM,KAAK,WAAW;gBAAE,OAAO,qBAAqB,CAAC;YACzD,IAAI,MAAM,KAAK,OAAO;gBAAE,OAAO,yBAAyB,CAAC;YACzD,OAAO,EAAE,CAAC;QACZ,CAAC,CAAC,CAAC;QAEH,iBAAY,GAAG,QAAQ,CAAC,GAAG,EAAE;YAC3B,MAAM,OAAO,GAAG,IAAI,CAAC,KAAK,EAAE,OAAO,CAAC;YACpC,IAAI,CAAC,OAAO,EAAE,MAAM;gBAAE,OAAO,EAAE,CAAC;YAEhC,MAAM,MAAM,GAAG,IAAI,CAAC,KAAK,EAAE,MAAM,IAAI,EAAE,CAAC;YAExC,IAAI,OAAO,CAAC,MAAM,CAAC,UAAU,CAAC,EAAE,CAAC;gBAC/B,OAAO,MAAM,CAAC,UAAU,CAAC,IAAI,0BAA0B,CAAC;YAC1D,CAAC;YACD,IAAI,OAAO,CAAC,MAAM,CAAC,WAAW,CAAC,EAAE,CAAC;gBAChC,MAAM,GAAG,GAAG,IAAI,CAAC,KAAK,EAAE,SAAS,IAAI,CAAC,CAAC;gBACvC,OAAO,MAAM,CAAC,WAAW,CAAC,IAAI,UAAU,GAAG,aAAa,CAAC;YAC3D,CAAC;YACD,IAAI,OAAO,CAAC,MAAM,CAAC,WAAW,CAAC,EAAE,CAAC;gBAChC,MAAM,GAAG,GAAG,IAAI,CAAC,KAAK,EAAE,SAAS,IAAI,EAAE,CAAC;gBACxC,OAAO,MAAM,CAAC,WAAW,CAAC,IAAI,UAAU,GAAG,aAAa,CAAC;YAC3D,CAAC;YACD,IAAI,OAAO,CAAC,MAAM,CAAC,SAAS,CAAC,EAAE,CAAC;gBAC9B,OAAO,MAAM,CAAC,SAAS,CAAC,IAAI,uCAAuC,CAAC;YACtE,CAAC;YAED,OAAO,EAAE,CAAC;QACZ,CAAC,CAAC,CAAC;KAoFJ;IAlFC,QAAQ;QACN,IAAI,CAAC,sBAAsB,EAAE,CAAC;IAChC,CAAC;IAED,WAAW;QACT,IAAI,CAAC,QAAQ,CAAC,IAAI,EAAE,CAAC;QACrB,IAAI,CAAC,QAAQ,CAAC,QAAQ,EAAE,CAAC;IAC3B,CAAC;IAED,OAAO;QACL,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;IAC3B,CAAC;IAED,MAAM;QACJ,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;QAC1B,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,aAAa,EAAE,CAAC;IACrC,CAAC;IAED,OAAO,CAAC,KAAkB;QACxB,MAAM,KAAK,GAAG,KAAK,CAAC,MAAM,CAAC,KAAK,IAAI,EAAE,CAAC;QACvC,8DAA8D;QAC9D,MAAM,UAAU,GAAG,IAAI,CAAC,iBAAiB,CAAC,KAAK,CAAC,CAAC;QAEjD,IAAI,UAAU,KAAK,KAAK,EAAE,CAAC;YACzB,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,QAAQ,CAAC,UAAU,EAAE,EAAE,SAAS,EAAE,KAAK,EAAE,CAAC,CAAC;QAChE,CAAC;QAED,6BAA6B;QAC7B,IAAI,IAAI,CAAC,KAAK,CAAC,iBAAiB,IAAI,IAAI,CAAC,aAAa,CAAC,UAAU,CAAC,EAAE,CAAC;YACnE,IAAI,CAAC,kBAAkB,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;QAC3C,CAAC;aAAM,IAAI,CAAC,IAAI,CAAC,aAAa,CAAC,UAAU,CAAC,IAAI,UAAU,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YACpE,IAAI,CAAC,kBAAkB,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;QACzC,CAAC;aAAM,CAAC;YACN,IAAI,CAAC,kBAAkB,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC;QACtC,CAAC;IACH,CAAC;IAEO,sBAAsB;QAC5B,IAAI,CAAC,IAAI,CAAC,KAAK,EAAE,iBAAiB;YAAE,OAAO;QAE3C,IAAI,CAAC,kBAAkB;aACpB,IAAI,CACH,YAAY,CAAC,IAAI,CAAC,KAAK,CAAC,YAAY,IAAI,GAAG,CAAC,EAC5C,oBAAoB,EAAE,EACtB,SAAS,CAAC,QAAQ,CAAC,EAAE;YACnB,IAAI,CAAC,QAAQ,IAAI,QAAQ,CAAC,MAAM,GAAG,CAAC,IAAI,CAAC,KAAK,CAAC,SAAS,IAAI,CAAC,CAAC,EAAE,CAAC;gBAC/D,OAAO,EAAE,CAAC,IAAI,CAAC,CAAC;YAClB,CAAC;YAED,IAAI,CAAC,kBAAkB,CAAC,GAAG,CAAC,UAAU,CAAC,CAAC;YAExC,OAAO,IAAI,CAAC,KAAK,CAAC,iBAAkB,CAAC,QAAQ,CAAC,CAAC,IAAI,CACjD,UAAU,CAAC,GAAG,EAAE,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC,CAC3B,CAAC;QACJ,CAAC,CAAC,EACF,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,CACzB;aACA,SAAS,CAAC,SAAS,CAAC,EAAE;YACrB,IAAI,SAAS,KAAK,IAAI,EAAE,CAAC;gBACvB,IAAI,CAAC,kBAAkB,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC;YACtC,CAAC;iBAAM,IAAI,SAAS,EAAE,CAAC;gBACrB,IAAI,CAAC,kBAAkB,CAAC,GAAG,CAAC,WAAW,CAAC,CAAC;YAC3C,CAAC;iBAAM,CAAC;gBACN,IAAI,CAAC,kBAAkB,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC;YACvC,CAAC;QACH,CAAC,CAAC,CAAC;IACP,CAAC;IAEO,iBAAiB,CAAC,KAAa;QACrC,OAAO,KAAK;aACT,WAAW,EAAE;aACb,OAAO,CAAC,KAAK,EAAE,EAAE,CAAC;aAClB,OAAO,CAAC,aAAa,EAAE,EAAE,CAAC,CAAC;IAChC,CAAC;IAEO,aAAa,CAAC,KAAa;QACjC,IAAI,CAAC,KAAK;YAAE,OAAO,KAAK,CAAC;QACzB,MAAM,MAAM,GAAG,IAAI,CAAC,KAAK,EAAE,SAAS,IAAI,CAAC,CAAC;QAC1C,MAAM,MAAM,GAAG,IAAI,CAAC,KAAK,EAAE,SAAS,IAAI,EAAE,CAAC;QAC3C,MAAM,OAAO,GAAG,cAAc,CAAC;QAC/B,OAAO,KAAK,CAAC,MAAM,IAAI,MAAM,IAAI,KAAK,CAAC,MAAM,IAAI,MAAM,IAAI,OAAO,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IACjF,CAAC;+GAzIU,sBAAsB;mGAAtB,sBAAsB,0GA1JvB;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAmDT,qhDApDS,YAAY,8BAAE,mBAAmB,6dAAE,QAAQ,8eAAE,UAAU,yGAAE,OAAO,2JAAE,OAAO;;4FA2JxE,sBAAsB;kBA9JlC,SAAS;+BACE,oBAAoB,cAClB,IAAI,WACP,CAAC,YAAY,EAAE,mBAAmB,EAAE,QAAQ,EAAE,UAAU,EAAE,OAAO,EAAE,OAAO,CAAC,YAC1E;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAmDT;8BAwGQ,KAAK;sBAAb,KAAK","sourcesContent":["import {\n  Component,\n  Input,\n  OnInit,\n  OnDestroy,\n  signal,\n  computed,\n  inject,\n} from '@angular/core';\nimport { CommonModule } from '@angular/common';\nimport { ReactiveFormsModule } from '@angular/forms';\nimport { IonInput, IonSpinner, IonIcon, IonText } from '@ionic/angular/standalone';\nimport { Subject, debounceTime, distinctUntilChanged, takeUntil, switchMap, of, catchError } from 'rxjs';\nimport { addIcons } from 'ionicons';\nimport { checkmarkCircle, closeCircle, alertCircle } from 'ionicons/icons';\nimport { UsernameInputMetadata, UsernameAvailabilityStatus } from './types';\n\naddIcons({ checkmarkCircle, closeCircle, alertCircle });\n\n/**\n * Username Input Component\n *\n * Input especializado para usernames/handles con:\n * - Prefijo '@' visual\n * - Validación de formato (alfanuméricos y _)\n * - Normalización automática (lowercase, sin espacios)\n * - Verificación de disponibilidad con debounce\n * - Estados visuales: available, taken, checking\n *\n * @example\n * <val-username-input\n *   [props]=\"{\n *     control: usernameControl,\n *     label: 'Nombre de usuario',\n *     placeholder: 'tu_username',\n *     checkAvailability: checkFn\n *   }\"\n * />\n */\n@Component({\n  selector: 'val-username-input',\n  standalone: true,\n  imports: [CommonModule, ReactiveFormsModule, IonInput, IonSpinner, IonIcon, IonText],\n  template: `\n    <div class=\"username-input-container\">\n      @if (props.label) {\n        <label class=\"username-label\">{{ props.label }}</label>\n      }\n\n      <div class=\"username-input-wrapper\" [class.focused]=\"isFocused()\" [class.error]=\"hasError()\">\n        <span class=\"username-prefix\">{{ props.prefix || '@' }}</span>\n        <ion-input\n          [formControl]=\"props.control\"\n          type=\"text\"\n          [placeholder]=\"props.placeholder || 'username'\"\n          [maxlength]=\"props.maxLength || 30\"\n          (ionFocus)=\"onFocus()\"\n          (ionBlur)=\"onBlur()\"\n          (ionInput)=\"onInput($event)\"\n          class=\"username-field\"\n        />\n\n        @if (props.showAvailability !== false) {\n          <div class=\"availability-indicator\">\n            @switch (availabilityStatus()) {\n              @case ('checking') {\n                <ion-spinner name=\"crescent\" class=\"checking-spinner\" />\n              }\n              @case ('available') {\n                <ion-icon name=\"checkmark-circle\" class=\"status-icon available\" />\n              }\n              @case ('taken') {\n                <ion-icon name=\"close-circle\" class=\"status-icon taken\" />\n              }\n              @case ('invalid') {\n                <ion-icon name=\"alert-circle\" class=\"status-icon invalid\" />\n              }\n            }\n          </div>\n        }\n      </div>\n\n      @if (showStatusMessage()) {\n        <ion-text [color]=\"statusColor()\" class=\"status-message\">\n          <small>{{ statusMessage() }}</small>\n        </ion-text>\n      }\n\n      @if (hasError() && errorMessage()) {\n        <ion-text color=\"danger\" class=\"error-message\">\n          <small>{{ errorMessage() }}</small>\n        </ion-text>\n      }\n    </div>\n  `,\n  styles: [`\n    .username-input-container {\n      margin-bottom: 1rem;\n    }\n\n    .username-label {\n      display: block;\n      font-size: 0.875rem;\n      font-weight: 500;\n      color: var(--ion-color-dark);\n      margin-bottom: 0.5rem;\n    }\n\n    .username-input-wrapper {\n      display: flex;\n      align-items: center;\n      background: var(--ion-background-color, #fff);\n      border: 1px solid var(--ion-border-color, #e0e0e0);\n      border-radius: 8px;\n      padding: 0 0.75rem;\n      transition: border-color 0.2s, box-shadow 0.2s;\n    }\n\n    .username-input-wrapper.focused {\n      border-color: var(--ion-color-primary);\n      box-shadow: 0 0 0 2px var(--ion-color-primary-tint);\n    }\n\n    .username-input-wrapper.error {\n      border-color: var(--ion-color-danger);\n    }\n\n    .username-prefix {\n      font-size: 1rem;\n      font-weight: 500;\n      color: var(--ion-color-medium);\n      user-select: none;\n    }\n\n    .username-field {\n      flex: 1;\n      --padding-start: 0.25rem;\n      --padding-end: 0;\n      --background: transparent;\n      font-size: 1rem;\n    }\n\n    .username-field::part(native) {\n      padding-left: 0.25rem;\n    }\n\n    .availability-indicator {\n      display: flex;\n      align-items: center;\n      justify-content: center;\n      width: 24px;\n      height: 24px;\n      margin-left: 0.5rem;\n    }\n\n    .checking-spinner {\n      width: 18px;\n      height: 18px;\n      --color: var(--ion-color-medium);\n    }\n\n    .status-icon {\n      font-size: 1.25rem;\n    }\n\n    .status-icon.available {\n      color: var(--ion-color-success);\n    }\n\n    .status-icon.taken {\n      color: var(--ion-color-danger);\n    }\n\n    .status-icon.invalid {\n      color: var(--ion-color-warning);\n    }\n\n    .status-message,\n    .error-message {\n      display: block;\n      margin-top: 0.25rem;\n      padding-left: 0.25rem;\n    }\n\n    /* Dark mode */\n    :host-context(body.dark) {\n      .username-input-wrapper {\n        background: var(--ion-color-step-50);\n        border-color: var(--ion-color-step-150);\n      }\n\n      .username-label {\n        color: var(--ion-color-light);\n      }\n    }\n  `],\n})\nexport class UsernameInputComponent implements OnInit, OnDestroy {\n  @Input() props!: UsernameInputMetadata;\n\n  private destroy$ = new Subject<void>();\n  private checkAvailability$ = new Subject<string>();\n\n  // Signals\n  isFocused = signal(false);\n  availabilityStatus = signal<UsernameAvailabilityStatus>('idle');\n\n  // Computed\n  hasError = computed(() => {\n    const control = this.props?.control;\n    return control?.touched && control?.invalid;\n  });\n\n  showStatusMessage = computed(() => {\n    const status = this.availabilityStatus();\n    return status === 'available' || status === 'taken';\n  });\n\n  statusColor = computed(() => {\n    return this.availabilityStatus() === 'available' ? 'success' : 'danger';\n  });\n\n  statusMessage = computed(() => {\n    const status = this.availabilityStatus();\n    if (status === 'available') return 'Username disponible';\n    if (status === 'taken') return 'Username ya está en uso';\n    return '';\n  });\n\n  errorMessage = computed(() => {\n    const control = this.props?.control;\n    if (!control?.errors) return '';\n\n    const errors = this.props?.errors || {};\n\n    if (control.errors['required']) {\n      return errors['required'] || 'El username es requerido';\n    }\n    if (control.errors['minlength']) {\n      const min = this.props?.minLength || 3;\n      return errors['minlength'] || `Mínimo ${min} caracteres`;\n    }\n    if (control.errors['maxlength']) {\n      const max = this.props?.maxLength || 30;\n      return errors['maxlength'] || `Máximo ${max} caracteres`;\n    }\n    if (control.errors['pattern']) {\n      return errors['pattern'] || 'Solo letras, números y guión bajo (_)';\n    }\n\n    return '';\n  });\n\n  ngOnInit(): void {\n    this.setupAvailabilityCheck();\n  }\n\n  ngOnDestroy(): void {\n    this.destroy$.next();\n    this.destroy$.complete();\n  }\n\n  onFocus(): void {\n    this.isFocused.set(true);\n  }\n\n  onBlur(): void {\n    this.isFocused.set(false);\n    this.props.control.markAsTouched();\n  }\n\n  onInput(event: CustomEvent): void {\n    const input = event.detail.value || '';\n    // Normalize: lowercase, remove spaces, only allow valid chars\n    const normalized = this.normalizeUsername(input);\n\n    if (normalized !== input) {\n      this.props.control.setValue(normalized, { emitEvent: false });\n    }\n\n    // Trigger availability check\n    if (this.props.checkAvailability && this.isValidFormat(normalized)) {\n      this.checkAvailability$.next(normalized);\n    } else if (!this.isValidFormat(normalized) && normalized.length > 0) {\n      this.availabilityStatus.set('invalid');\n    } else {\n      this.availabilityStatus.set('idle');\n    }\n  }\n\n  private setupAvailabilityCheck(): void {\n    if (!this.props?.checkAvailability) return;\n\n    this.checkAvailability$\n      .pipe(\n        debounceTime(this.props.debounceTime || 500),\n        distinctUntilChanged(),\n        switchMap(username => {\n          if (!username || username.length < (this.props.minLength || 3)) {\n            return of(null);\n          }\n\n          this.availabilityStatus.set('checking');\n\n          return this.props.checkAvailability!(username).pipe(\n            catchError(() => of(null))\n          );\n        }),\n        takeUntil(this.destroy$)\n      )\n      .subscribe(available => {\n        if (available === null) {\n          this.availabilityStatus.set('idle');\n        } else if (available) {\n          this.availabilityStatus.set('available');\n        } else {\n          this.availabilityStatus.set('taken');\n        }\n      });\n  }\n\n  private normalizeUsername(value: string): string {\n    return value\n      .toLowerCase()\n      .replace(/\\s/g, '')\n      .replace(/[^a-z0-9_]/g, '');\n  }\n\n  private isValidFormat(value: string): boolean {\n    if (!value) return false;\n    const minLen = this.props?.minLength || 3;\n    const maxLen = this.props?.maxLength || 30;\n    const pattern = /^[a-z0-9_]+$/;\n    return value.length >= minLen && value.length <= maxLen && pattern.test(value);\n  }\n}\n"]}