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.
- package/esm2022/lib/components/molecules/action-card/action-card.component.mjs +298 -0
- package/esm2022/lib/components/molecules/action-card/types.mjs +11 -0
- package/esm2022/lib/components/molecules/linked-providers/linked-providers.component.mjs +236 -0
- package/esm2022/lib/components/molecules/linked-providers/types.mjs +27 -0
- package/esm2022/lib/components/molecules/stats-card/stats-card.component.mjs +33 -3
- package/esm2022/lib/components/molecules/stats-card/types.mjs +1 -1
- package/esm2022/lib/components/molecules/username-input/types.mjs +2 -0
- package/esm2022/lib/components/molecules/username-input/username-input.component.mjs +260 -0
- package/esm2022/lib/components/templates/docs-page/docs-page.component.mjs +3 -3
- package/esm2022/lib/services/auth/auth.service.mjs +24 -1
- package/esm2022/lib/services/auth/types.mjs +1 -1
- package/esm2022/public-api.mjs +7 -1
- package/fesm2022/valtech-components.mjs +904 -46
- package/fesm2022/valtech-components.mjs.map +1 -1
- package/lib/components/molecules/action-card/action-card.component.d.ts +90 -0
- package/lib/components/molecules/action-card/types.d.ts +83 -0
- package/lib/components/molecules/linked-providers/linked-providers.component.d.ts +30 -0
- package/lib/components/molecules/linked-providers/types.d.ts +38 -0
- package/lib/components/molecules/stats-card/types.d.ts +17 -0
- package/lib/components/molecules/username-input/types.d.ts +34 -0
- package/lib/components/molecules/username-input/username-input.component.d.ts +45 -0
- package/lib/services/auth/auth.service.d.ts +11 -1
- package/lib/services/auth/types.d.ts +56 -0
- package/package.json +1 -1
- 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,
|
|
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":"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","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"]}
|