valtech-components 2.0.628 → 2.0.629

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.
@@ -0,0 +1,298 @@
1
+ import { CommonModule } from '@angular/common';
2
+ import { Component, computed, EventEmitter, inject, input, Output, } from '@angular/core';
3
+ import { RouterLink } from '@angular/router';
4
+ import { IonIcon, IonRippleEffect } from '@ionic/angular/standalone';
5
+ import { addIcons } from 'ionicons';
6
+ import { chevronForwardOutline } from 'ionicons/icons';
7
+ import { I18nService } from '../../../services/i18n';
8
+ import { NavigationService } from '../../../services/navigation.service';
9
+ import { ACTION_CARD_DEFAULTS, } from './types';
10
+ import * as i0 from "@angular/core";
11
+ addIcons({ chevronForwardOutline });
12
+ const IONIC_COLORS = [
13
+ 'primary', 'secondary', 'tertiary', 'success',
14
+ 'warning', 'danger', 'light', 'medium', 'dark'
15
+ ];
16
+ /**
17
+ * val-action-card
18
+ *
19
+ * A clickable card component with icon, title, description, and optional badge.
20
+ * Supports multiple icon formats: Ionicons, SVG paths, and image URLs.
21
+ *
22
+ * @example Basic usage with Ionicon
23
+ * ```html
24
+ * <val-action-card
25
+ * [props]="{
26
+ * icon: { ionicon: 'settings-outline' },
27
+ * title: 'Settings',
28
+ * description: 'Manage your preferences'
29
+ * }"
30
+ * (cardClick)="onCardClick($event)"
31
+ * />
32
+ * ```
33
+ *
34
+ * @example With routerLink navigation
35
+ * ```html
36
+ * <val-action-card
37
+ * [props]="{
38
+ * icon: { ionicon: 'person-outline', color: 'primary' },
39
+ * title: 'Profile',
40
+ * description: 'View and edit your profile',
41
+ * routerLink: '/settings/profile',
42
+ * showChevron: true
43
+ * }"
44
+ * />
45
+ * ```
46
+ *
47
+ * @example With custom SVG icon
48
+ * ```html
49
+ * <val-action-card
50
+ * [props]="{
51
+ * icon: { svgPath: 'M12 2L2 7l10 5 10-5-10-5zM2 17l10 5 10-5M2 12l10 5 10-5' },
52
+ * title: 'Custom Feature',
53
+ * description: 'A feature with custom SVG icon',
54
+ * badge: { text: 'NEW', color: 'light', backgroundColor: 'primary' }
55
+ * }"
56
+ * />
57
+ * ```
58
+ */
59
+ export class ActionCardComponent {
60
+ constructor() {
61
+ this.i18n = inject(I18nService);
62
+ this.navigation = inject(NavigationService);
63
+ /** Component configuration */
64
+ this.props = input({});
65
+ /** Event emitted when card is clicked */
66
+ this.cardClick = new EventEmitter();
67
+ /** Merged configuration with defaults */
68
+ this.config = computed(() => ({
69
+ ...ACTION_CARD_DEFAULTS,
70
+ ...this.props(),
71
+ }));
72
+ }
73
+ /** Get title with i18n support */
74
+ getTitle() {
75
+ const cfg = this.config();
76
+ if (cfg.i18nNamespace && cfg.titleKey) {
77
+ this.i18n.lang(); // Track language changes for reactivity
78
+ return this.i18n.t(cfg.titleKey, cfg.i18nNamespace);
79
+ }
80
+ return cfg.title || '';
81
+ }
82
+ /** Get description with i18n support */
83
+ getDescription() {
84
+ const cfg = this.config();
85
+ if (cfg.i18nNamespace && cfg.descriptionKey) {
86
+ this.i18n.lang(); // Track language changes for reactivity
87
+ return this.i18n.t(cfg.descriptionKey, cfg.i18nNamespace);
88
+ }
89
+ return cfg.description || '';
90
+ }
91
+ /** Resolve color to CSS value */
92
+ resolveColor(color) {
93
+ if (!color)
94
+ return null;
95
+ if (IONIC_COLORS.includes(color)) {
96
+ return `var(--ion-color-${color})`;
97
+ }
98
+ return color;
99
+ }
100
+ getBackgroundColor() {
101
+ return this.resolveColor(this.config().backgroundColor);
102
+ }
103
+ getBorderColor() {
104
+ return this.resolveColor(this.config().borderColor) || 'var(--ion-color-light-shade)';
105
+ }
106
+ getIconColor() {
107
+ return this.resolveColor(this.config().icon?.color) || 'var(--ion-color-primary)';
108
+ }
109
+ getIconBackgroundColor() {
110
+ const bg = this.config().icon?.backgroundColor;
111
+ if (!bg)
112
+ return 'rgba(var(--ion-color-primary-rgb), 0.1)';
113
+ return this.resolveColor(bg);
114
+ }
115
+ getBadgeColor() {
116
+ return this.resolveColor(this.config().badge?.color) || 'white';
117
+ }
118
+ getBadgeBackgroundColor() {
119
+ return this.resolveColor(this.config().badge?.backgroundColor) || 'var(--ion-color-primary)';
120
+ }
121
+ /** Handle card click */
122
+ handleClick(event) {
123
+ const cfg = this.config();
124
+ if (cfg.disabled) {
125
+ event.preventDefault();
126
+ event.stopPropagation();
127
+ return;
128
+ }
129
+ // Emit click event
130
+ this.cardClick.emit({
131
+ token: cfg.token,
132
+ navigated: !!cfg.routerLink || !!cfg.href,
133
+ });
134
+ // Handle external URL
135
+ if (cfg.href) {
136
+ this.navigation.openInNewTab(cfg.href);
137
+ }
138
+ }
139
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: ActionCardComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
140
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: ActionCardComponent, isStandalone: true, selector: "val-action-card", inputs: { props: { classPropertyName: "props", publicName: "props", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { cardClick: "cardClick" }, ngImport: i0, template: `
141
+ <article
142
+ class="action-card"
143
+ [class.action-card--small]="config().size === 'small'"
144
+ [class.action-card--medium]="config().size === 'medium'"
145
+ [class.action-card--large]="config().size === 'large'"
146
+ [class.action-card--bordered]="config().bordered"
147
+ [class.action-card--shadowed]="config().shadowed"
148
+ [class.action-card--disabled]="config().disabled"
149
+ [class.action-card--clickable]="!config().disabled"
150
+ [style.--card-bg]="getBackgroundColor()"
151
+ [style.--card-border-color]="getBorderColor()"
152
+ [routerLink]="config().disabled ? null : config().routerLink"
153
+ (click)="handleClick($event)"
154
+ [attr.tabindex]="config().disabled ? -1 : 0"
155
+ [attr.role]="'button'"
156
+ [attr.aria-disabled]="config().disabled"
157
+ >
158
+ <ion-ripple-effect></ion-ripple-effect>
159
+
160
+ <!-- Badge (top-right corner) -->
161
+ @if (config().badge) {
162
+ <span
163
+ class="action-card__badge"
164
+ [style.color]="getBadgeColor()"
165
+ [style.background-color]="getBadgeBackgroundColor()"
166
+ >
167
+ {{ config().badge!.text }}
168
+ </span>
169
+ }
170
+
171
+ <!-- Icon Container -->
172
+ <div
173
+ class="action-card__icon"
174
+ [style.color]="getIconColor()"
175
+ [style.background-color]="getIconBackgroundColor()"
176
+ >
177
+ @if (config().icon?.ionicon) {
178
+ <ion-icon [name]="config().icon!.ionicon!"></ion-icon>
179
+ } @else if (config().icon?.svgPath) {
180
+ <svg
181
+ viewBox="0 0 24 24"
182
+ fill="none"
183
+ stroke="currentColor"
184
+ stroke-width="1.5"
185
+ stroke-linecap="round"
186
+ stroke-linejoin="round"
187
+ >
188
+ <path [attr.d]="config().icon!.svgPath" />
189
+ </svg>
190
+ } @else if (config().icon?.imageUrl) {
191
+ <img
192
+ [src]="config().icon!.imageUrl"
193
+ [alt]="getTitle()"
194
+ class="action-card__icon-image"
195
+ />
196
+ }
197
+ </div>
198
+
199
+ <!-- Content -->
200
+ <div class="action-card__content">
201
+ <h3 class="action-card__title">{{ getTitle() }}</h3>
202
+ @if (getDescription()) {
203
+ <p class="action-card__description">{{ getDescription() }}</p>
204
+ }
205
+ </div>
206
+
207
+ <!-- Chevron (optional) -->
208
+ @if (config().showChevron && !config().disabled) {
209
+ <ion-icon
210
+ name="chevron-forward-outline"
211
+ class="action-card__chevron"
212
+ ></ion-icon>
213
+ }
214
+ </article>
215
+ `, isInline: true, styles: [":host{display:block}.action-card{position:relative;display:flex;align-items:center;gap:1rem;padding:1rem;background:var(--card-bg, var(--ion-card-background, var(--ion-background-color)));border-radius:12px;cursor:pointer;transition:transform .2s ease,box-shadow .2s ease,background-color .2s ease;text-decoration:none;overflow:hidden;--ripple-color: var(--ion-color-primary)}.action-card--small{padding:.75rem;gap:.75rem}.action-card--small .action-card__icon{width:36px;height:36px;font-size:18px;border-radius:8px}.action-card--small .action-card__icon svg{width:18px;height:18px}.action-card--small .action-card__title{font-size:.9rem}.action-card--small .action-card__description{font-size:.8rem}.action-card--medium{padding:1rem;gap:1rem}.action-card--medium .action-card__icon{width:48px;height:48px;font-size:24px;border-radius:10px}.action-card--medium .action-card__icon svg{width:24px;height:24px}.action-card--medium .action-card__title{font-size:1rem}.action-card--medium .action-card__description{font-size:.875rem}.action-card--large{padding:1.25rem;gap:1.25rem}.action-card--large .action-card__icon{width:56px;height:56px;font-size:28px;border-radius:12px}.action-card--large .action-card__icon svg{width:28px;height:28px}.action-card--large .action-card__title{font-size:1.1rem}.action-card--large .action-card__description{font-size:.9rem}.action-card--bordered{border:1px solid var(--card-border-color, var(--ion-color-light-shade))}.action-card--shadowed{box-shadow:0 2px 8px #00000014}.action-card--clickable:hover{transform:translateY(-2px);box-shadow:0 4px 16px #0000001f}.action-card--clickable:focus-visible{outline:2px solid var(--ion-color-primary);outline-offset:2px}.action-card--clickable:active{transform:translateY(0)}.action-card--disabled{opacity:.5;cursor:not-allowed;pointer-events:none}.action-card__badge{position:absolute;top:8px;right:8px;padding:2px 8px;font-size:.7rem;font-weight:600;text-transform:uppercase;letter-spacing:.5px;border-radius:10px;z-index:1}.action-card__icon{flex-shrink:0;display:flex;align-items:center;justify-content:center}.action-card__icon ion-icon{font-size:inherit}.action-card__icon svg{display:block}.action-card__icon-image{width:100%;height:100%;object-fit:cover;border-radius:8px}.action-card__content{flex:1;min-width:0}.action-card__title{margin:0 0 .25rem;font-weight:600;color:var(--ion-text-color);line-height:1.3}.action-card__description{margin:0;color:var(--ion-color-medium);line-height:1.4}.action-card__chevron{flex-shrink:0;font-size:1.25rem;color:var(--ion-color-medium);transition:transform .2s ease}.action-card--clickable:hover .action-card__chevron{transform:translate(4px)}@media (prefers-color-scheme: dark){.action-card--shadowed{box-shadow:0 2px 8px #0000004d}.action-card--clickable:hover{box-shadow:0 4px 16px #0006}}:host-context(.dark) .action-card--shadowed,:host-context(body.dark) .action-card--shadowed,:host-context([data-theme=dark]) .action-card--shadowed{box-shadow:0 2px 8px #0000004d}:host-context(.dark) .action-card--clickable:hover,:host-context(body.dark) .action-card--clickable:hover,:host-context([data-theme=dark]) .action-card--clickable:hover{box-shadow:0 4px 16px #0006}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: RouterLink, selector: "[routerLink]", inputs: ["target", "queryParams", "fragment", "queryParamsHandling", "state", "info", "relativeTo", "preserveFragment", "skipLocationChange", "replaceUrl", "routerLink"] }, { kind: "component", type: IonIcon, selector: "ion-icon", inputs: ["color", "flipRtl", "icon", "ios", "lazy", "md", "mode", "name", "sanitize", "size", "src"] }, { kind: "component", type: IonRippleEffect, selector: "ion-ripple-effect", inputs: ["type"] }] }); }
216
+ }
217
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: ActionCardComponent, decorators: [{
218
+ type: Component,
219
+ args: [{ selector: 'val-action-card', standalone: true, imports: [CommonModule, RouterLink, IonIcon, IonRippleEffect], template: `
220
+ <article
221
+ class="action-card"
222
+ [class.action-card--small]="config().size === 'small'"
223
+ [class.action-card--medium]="config().size === 'medium'"
224
+ [class.action-card--large]="config().size === 'large'"
225
+ [class.action-card--bordered]="config().bordered"
226
+ [class.action-card--shadowed]="config().shadowed"
227
+ [class.action-card--disabled]="config().disabled"
228
+ [class.action-card--clickable]="!config().disabled"
229
+ [style.--card-bg]="getBackgroundColor()"
230
+ [style.--card-border-color]="getBorderColor()"
231
+ [routerLink]="config().disabled ? null : config().routerLink"
232
+ (click)="handleClick($event)"
233
+ [attr.tabindex]="config().disabled ? -1 : 0"
234
+ [attr.role]="'button'"
235
+ [attr.aria-disabled]="config().disabled"
236
+ >
237
+ <ion-ripple-effect></ion-ripple-effect>
238
+
239
+ <!-- Badge (top-right corner) -->
240
+ @if (config().badge) {
241
+ <span
242
+ class="action-card__badge"
243
+ [style.color]="getBadgeColor()"
244
+ [style.background-color]="getBadgeBackgroundColor()"
245
+ >
246
+ {{ config().badge!.text }}
247
+ </span>
248
+ }
249
+
250
+ <!-- Icon Container -->
251
+ <div
252
+ class="action-card__icon"
253
+ [style.color]="getIconColor()"
254
+ [style.background-color]="getIconBackgroundColor()"
255
+ >
256
+ @if (config().icon?.ionicon) {
257
+ <ion-icon [name]="config().icon!.ionicon!"></ion-icon>
258
+ } @else if (config().icon?.svgPath) {
259
+ <svg
260
+ viewBox="0 0 24 24"
261
+ fill="none"
262
+ stroke="currentColor"
263
+ stroke-width="1.5"
264
+ stroke-linecap="round"
265
+ stroke-linejoin="round"
266
+ >
267
+ <path [attr.d]="config().icon!.svgPath" />
268
+ </svg>
269
+ } @else if (config().icon?.imageUrl) {
270
+ <img
271
+ [src]="config().icon!.imageUrl"
272
+ [alt]="getTitle()"
273
+ class="action-card__icon-image"
274
+ />
275
+ }
276
+ </div>
277
+
278
+ <!-- Content -->
279
+ <div class="action-card__content">
280
+ <h3 class="action-card__title">{{ getTitle() }}</h3>
281
+ @if (getDescription()) {
282
+ <p class="action-card__description">{{ getDescription() }}</p>
283
+ }
284
+ </div>
285
+
286
+ <!-- Chevron (optional) -->
287
+ @if (config().showChevron && !config().disabled) {
288
+ <ion-icon
289
+ name="chevron-forward-outline"
290
+ class="action-card__chevron"
291
+ ></ion-icon>
292
+ }
293
+ </article>
294
+ `, styles: [":host{display:block}.action-card{position:relative;display:flex;align-items:center;gap:1rem;padding:1rem;background:var(--card-bg, var(--ion-card-background, var(--ion-background-color)));border-radius:12px;cursor:pointer;transition:transform .2s ease,box-shadow .2s ease,background-color .2s ease;text-decoration:none;overflow:hidden;--ripple-color: var(--ion-color-primary)}.action-card--small{padding:.75rem;gap:.75rem}.action-card--small .action-card__icon{width:36px;height:36px;font-size:18px;border-radius:8px}.action-card--small .action-card__icon svg{width:18px;height:18px}.action-card--small .action-card__title{font-size:.9rem}.action-card--small .action-card__description{font-size:.8rem}.action-card--medium{padding:1rem;gap:1rem}.action-card--medium .action-card__icon{width:48px;height:48px;font-size:24px;border-radius:10px}.action-card--medium .action-card__icon svg{width:24px;height:24px}.action-card--medium .action-card__title{font-size:1rem}.action-card--medium .action-card__description{font-size:.875rem}.action-card--large{padding:1.25rem;gap:1.25rem}.action-card--large .action-card__icon{width:56px;height:56px;font-size:28px;border-radius:12px}.action-card--large .action-card__icon svg{width:28px;height:28px}.action-card--large .action-card__title{font-size:1.1rem}.action-card--large .action-card__description{font-size:.9rem}.action-card--bordered{border:1px solid var(--card-border-color, var(--ion-color-light-shade))}.action-card--shadowed{box-shadow:0 2px 8px #00000014}.action-card--clickable:hover{transform:translateY(-2px);box-shadow:0 4px 16px #0000001f}.action-card--clickable:focus-visible{outline:2px solid var(--ion-color-primary);outline-offset:2px}.action-card--clickable:active{transform:translateY(0)}.action-card--disabled{opacity:.5;cursor:not-allowed;pointer-events:none}.action-card__badge{position:absolute;top:8px;right:8px;padding:2px 8px;font-size:.7rem;font-weight:600;text-transform:uppercase;letter-spacing:.5px;border-radius:10px;z-index:1}.action-card__icon{flex-shrink:0;display:flex;align-items:center;justify-content:center}.action-card__icon ion-icon{font-size:inherit}.action-card__icon svg{display:block}.action-card__icon-image{width:100%;height:100%;object-fit:cover;border-radius:8px}.action-card__content{flex:1;min-width:0}.action-card__title{margin:0 0 .25rem;font-weight:600;color:var(--ion-text-color);line-height:1.3}.action-card__description{margin:0;color:var(--ion-color-medium);line-height:1.4}.action-card__chevron{flex-shrink:0;font-size:1.25rem;color:var(--ion-color-medium);transition:transform .2s ease}.action-card--clickable:hover .action-card__chevron{transform:translate(4px)}@media (prefers-color-scheme: dark){.action-card--shadowed{box-shadow:0 2px 8px #0000004d}.action-card--clickable:hover{box-shadow:0 4px 16px #0006}}:host-context(.dark) .action-card--shadowed,:host-context(body.dark) .action-card--shadowed,:host-context([data-theme=dark]) .action-card--shadowed{box-shadow:0 2px 8px #0000004d}:host-context(.dark) .action-card--clickable:hover,:host-context(body.dark) .action-card--clickable:hover,:host-context([data-theme=dark]) .action-card--clickable:hover{box-shadow:0 4px 16px #0006}\n"] }]
295
+ }], propDecorators: { cardClick: [{
296
+ type: Output
297
+ }] } });
298
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"action-card.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/molecules/action-card/action-card.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport {\n  Component,\n  computed,\n  EventEmitter,\n  inject,\n  input,\n  Output,\n} from '@angular/core';\nimport { RouterLink } from '@angular/router';\nimport { IonIcon, IonRippleEffect } from '@ionic/angular/standalone';\nimport { addIcons } from 'ionicons';\nimport { chevronForwardOutline } from 'ionicons/icons';\nimport { I18nService } from '../../../services/i18n';\nimport { NavigationService } from '../../../services/navigation.service';\nimport {\n  ActionCardMetadata,\n  ActionCardClickEvent,\n  ACTION_CARD_DEFAULTS,\n} from './types';\n\naddIcons({ chevronForwardOutline });\n\nconst IONIC_COLORS = [\n  'primary', 'secondary', 'tertiary', 'success',\n  'warning', 'danger', 'light', 'medium', 'dark'\n];\n\n/**\n * val-action-card\n *\n * A clickable card component with icon, title, description, and optional badge.\n * Supports multiple icon formats: Ionicons, SVG paths, and image URLs.\n *\n * @example Basic usage with Ionicon\n * ```html\n * <val-action-card\n *   [props]=\"{\n *     icon: { ionicon: 'settings-outline' },\n *     title: 'Settings',\n *     description: 'Manage your preferences'\n *   }\"\n *   (cardClick)=\"onCardClick($event)\"\n * />\n * ```\n *\n * @example With routerLink navigation\n * ```html\n * <val-action-card\n *   [props]=\"{\n *     icon: { ionicon: 'person-outline', color: 'primary' },\n *     title: 'Profile',\n *     description: 'View and edit your profile',\n *     routerLink: '/settings/profile',\n *     showChevron: true\n *   }\"\n * />\n * ```\n *\n * @example With custom SVG icon\n * ```html\n * <val-action-card\n *   [props]=\"{\n *     icon: { svgPath: 'M12 2L2 7l10 5 10-5-10-5zM2 17l10 5 10-5M2 12l10 5 10-5' },\n *     title: 'Custom Feature',\n *     description: 'A feature with custom SVG icon',\n *     badge: { text: 'NEW', color: 'light', backgroundColor: 'primary' }\n *   }\"\n * />\n * ```\n */\n@Component({\n  selector: 'val-action-card',\n  standalone: true,\n  imports: [CommonModule, RouterLink, IonIcon, IonRippleEffect],\n  template: `\n    <article\n      class=\"action-card\"\n      [class.action-card--small]=\"config().size === 'small'\"\n      [class.action-card--medium]=\"config().size === 'medium'\"\n      [class.action-card--large]=\"config().size === 'large'\"\n      [class.action-card--bordered]=\"config().bordered\"\n      [class.action-card--shadowed]=\"config().shadowed\"\n      [class.action-card--disabled]=\"config().disabled\"\n      [class.action-card--clickable]=\"!config().disabled\"\n      [style.--card-bg]=\"getBackgroundColor()\"\n      [style.--card-border-color]=\"getBorderColor()\"\n      [routerLink]=\"config().disabled ? null : config().routerLink\"\n      (click)=\"handleClick($event)\"\n      [attr.tabindex]=\"config().disabled ? -1 : 0\"\n      [attr.role]=\"'button'\"\n      [attr.aria-disabled]=\"config().disabled\"\n    >\n      <ion-ripple-effect></ion-ripple-effect>\n\n      <!-- Badge (top-right corner) -->\n      @if (config().badge) {\n        <span\n          class=\"action-card__badge\"\n          [style.color]=\"getBadgeColor()\"\n          [style.background-color]=\"getBadgeBackgroundColor()\"\n        >\n          {{ config().badge!.text }}\n        </span>\n      }\n\n      <!-- Icon Container -->\n      <div\n        class=\"action-card__icon\"\n        [style.color]=\"getIconColor()\"\n        [style.background-color]=\"getIconBackgroundColor()\"\n      >\n        @if (config().icon?.ionicon) {\n          <ion-icon [name]=\"config().icon!.ionicon!\"></ion-icon>\n        } @else if (config().icon?.svgPath) {\n          <svg\n            viewBox=\"0 0 24 24\"\n            fill=\"none\"\n            stroke=\"currentColor\"\n            stroke-width=\"1.5\"\n            stroke-linecap=\"round\"\n            stroke-linejoin=\"round\"\n          >\n            <path [attr.d]=\"config().icon!.svgPath\" />\n          </svg>\n        } @else if (config().icon?.imageUrl) {\n          <img\n            [src]=\"config().icon!.imageUrl\"\n            [alt]=\"getTitle()\"\n            class=\"action-card__icon-image\"\n          />\n        }\n      </div>\n\n      <!-- Content -->\n      <div class=\"action-card__content\">\n        <h3 class=\"action-card__title\">{{ getTitle() }}</h3>\n        @if (getDescription()) {\n          <p class=\"action-card__description\">{{ getDescription() }}</p>\n        }\n      </div>\n\n      <!-- Chevron (optional) -->\n      @if (config().showChevron && !config().disabled) {\n        <ion-icon\n          name=\"chevron-forward-outline\"\n          class=\"action-card__chevron\"\n        ></ion-icon>\n      }\n    </article>\n  `,\n  styleUrls: ['./action-card.component.scss'],\n})\nexport class ActionCardComponent {\n  private i18n = inject(I18nService);\n  private navigation = inject(NavigationService);\n\n  /** Component configuration */\n  readonly props = input<Partial<ActionCardMetadata>>({});\n\n  /** Event emitted when card is clicked */\n  @Output() cardClick = new EventEmitter<ActionCardClickEvent>();\n\n  /** Merged configuration with defaults */\n  config = computed(() => ({\n    ...ACTION_CARD_DEFAULTS,\n    ...this.props(),\n  }));\n\n  /** Get title with i18n support */\n  getTitle(): string {\n    const cfg = this.config();\n    if (cfg.i18nNamespace && cfg.titleKey) {\n      this.i18n.lang(); // Track language changes for reactivity\n      return this.i18n.t(cfg.titleKey, cfg.i18nNamespace);\n    }\n    return cfg.title || '';\n  }\n\n  /** Get description with i18n support */\n  getDescription(): string {\n    const cfg = this.config();\n    if (cfg.i18nNamespace && cfg.descriptionKey) {\n      this.i18n.lang(); // Track language changes for reactivity\n      return this.i18n.t(cfg.descriptionKey, cfg.i18nNamespace);\n    }\n    return cfg.description || '';\n  }\n\n  /** Resolve color to CSS value */\n  private resolveColor(color?: string): string | null {\n    if (!color) return null;\n    if (IONIC_COLORS.includes(color)) {\n      return `var(--ion-color-${color})`;\n    }\n    return color;\n  }\n\n  getBackgroundColor(): string | null {\n    return this.resolveColor(this.config().backgroundColor as string);\n  }\n\n  getBorderColor(): string | null {\n    return this.resolveColor(this.config().borderColor as string) || 'var(--ion-color-light-shade)';\n  }\n\n  getIconColor(): string | null {\n    return this.resolveColor(this.config().icon?.color as string) || 'var(--ion-color-primary)';\n  }\n\n  getIconBackgroundColor(): string | null {\n    const bg = this.config().icon?.backgroundColor;\n    if (!bg) return 'rgba(var(--ion-color-primary-rgb), 0.1)';\n    return this.resolveColor(bg as string);\n  }\n\n  getBadgeColor(): string | null {\n    return this.resolveColor(this.config().badge?.color as string) || 'white';\n  }\n\n  getBadgeBackgroundColor(): string | null {\n    return this.resolveColor(this.config().badge?.backgroundColor as string) || 'var(--ion-color-primary)';\n  }\n\n  /** Handle card click */\n  handleClick(event: MouseEvent): void {\n    const cfg = this.config();\n\n    if (cfg.disabled) {\n      event.preventDefault();\n      event.stopPropagation();\n      return;\n    }\n\n    // Emit click event\n    this.cardClick.emit({\n      token: cfg.token,\n      navigated: !!cfg.routerLink || !!cfg.href,\n    });\n\n    // Handle external URL\n    if (cfg.href) {\n      this.navigation.openInNewTab(cfg.href);\n    }\n  }\n}\n"]}
@@ -0,0 +1,11 @@
1
+ /**
2
+ * Default values for ActionCardMetadata
3
+ */
4
+ export const ACTION_CARD_DEFAULTS = {
5
+ size: 'medium',
6
+ bordered: false,
7
+ shadowed: true,
8
+ disabled: false,
9
+ showChevron: false,
10
+ };
11
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL2NvbXBvbmVudHMvbW9sZWN1bGVzL2FjdGlvbi1jYXJkL3R5cGVzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQTBGQTs7R0FFRztBQUNILE1BQU0sQ0FBQyxNQUFNLG9CQUFvQixHQUU3QjtJQUNGLElBQUksRUFBRSxRQUFRO0lBQ2QsUUFBUSxFQUFFLEtBQUs7SUFDZixRQUFRLEVBQUUsSUFBSTtJQUNkLFFBQVEsRUFBRSxLQUFLO0lBQ2YsV0FBVyxFQUFFLEtBQUs7Q0FDbkIsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENvbG9yIH0gZnJvbSAnQGlvbmljL2NvcmUnO1xuXG4vKipcbiAqIFNpemUgb3B0aW9ucyBmb3IgYWN0aW9uLWNhcmRcbiAqL1xuZXhwb3J0IHR5cGUgQWN0aW9uQ2FyZFNpemUgPSAnc21hbGwnIHwgJ21lZGl1bScgfCAnbGFyZ2UnO1xuXG4vKipcbiAqIEJhZGdlIGNvbmZpZ3VyYXRpb24gZm9yIGFjdGlvbi1jYXJkXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgQWN0aW9uQ2FyZEJhZGdlIHtcbiAgLyoqIEJhZGdlIHRleHQgKi9cbiAgdGV4dDogc3RyaW5nO1xuICAvKiogQmFkZ2UgY29sb3IgKElvbmljIGNvbG9yIG5hbWUgb3IgQ1NTIGNvbG9yKSAqL1xuICBjb2xvcj86IENvbG9yIHwgc3RyaW5nO1xuICAvKiogQmFkZ2UgYmFja2dyb3VuZCBjb2xvciAqL1xuICBiYWNrZ3JvdW5kQ29sb3I/OiBDb2xvciB8IHN0cmluZztcbn1cblxuLyoqXG4gKiBJY29uIGNvbmZpZ3VyYXRpb24gLSBzdXBwb3J0cyBtdWx0aXBsZSBpY29uIHNvdXJjZXNcbiAqL1xuZXhwb3J0IGludGVyZmFjZSBBY3Rpb25DYXJkSWNvbiB7XG4gIC8qKiBJb25pY29uIG5hbWUgKGUuZy4sICdzZXR0aW5ncy1vdXRsaW5lJywgJ2hvbWUnKSAqL1xuICBpb25pY29uPzogc3RyaW5nO1xuICAvKiogU1ZHIHBhdGggZGF0YSBmb3IgY3VzdG9tIGljb25zICovXG4gIHN2Z1BhdGg/OiBzdHJpbmc7XG4gIC8qKiBJbWFnZSBVUkwgZm9yIGN1c3RvbSBpbWFnZXMgKi9cbiAgaW1hZ2VVcmw/OiBzdHJpbmc7XG4gIC8qKiBJY29uIGNvbG9yIChJb25pYyBjb2xvciBuYW1lIG9yIENTUyBjb2xvcikgKi9cbiAgY29sb3I/OiBDb2xvciB8IHN0cmluZztcbiAgLyoqIEljb24gYmFja2dyb3VuZCBjb2xvciAqL1xuICBiYWNrZ3JvdW5kQ29sb3I/OiBDb2xvciB8IHN0cmluZztcbn1cblxuLyoqXG4gKiBDbGljayBldmVudCBlbWl0dGVkIGJ5IGFjdGlvbi1jYXJkXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgQWN0aW9uQ2FyZENsaWNrRXZlbnQge1xuICAvKiogVG9rZW4gaWRlbnRpZmllciBmb3IgdGhlIGNhcmQgKi9cbiAgdG9rZW4/OiBzdHJpbmc7XG4gIC8qKiBXaGV0aGVyIG5hdmlnYXRpb24gd2FzIHRyaWdnZXJlZCAoaWYgcm91dGVyTGluayB3YXMgc2V0KSAqL1xuICBuYXZpZ2F0ZWQ/OiBib29sZWFuO1xufVxuXG4vKipcbiAqIE1ldGFkYXRhIGZvciB2YWwtYWN0aW9uLWNhcmQgY29tcG9uZW50XG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgQWN0aW9uQ2FyZE1ldGFkYXRhIHtcbiAgLyoqIFVuaXF1ZSB0b2tlbiBmb3IgaWRlbnRpZmljYXRpb24gKi9cbiAgdG9rZW4/OiBzdHJpbmc7XG5cbiAgLyoqIEljb24gY29uZmlndXJhdGlvbiBvYmplY3QgKi9cbiAgaWNvbj86IEFjdGlvbkNhcmRJY29uO1xuXG4gIC8qKiBDYXJkIHRpdGxlIChzdGF0aWMgdGV4dCkgKi9cbiAgdGl0bGU6IHN0cmluZztcbiAgLyoqIGkxOG4ga2V5IGZvciB0aXRsZSAqL1xuICB0aXRsZUtleT86IHN0cmluZztcbiAgLyoqIENhcmQgZGVzY3JpcHRpb24gKHN0YXRpYyB0ZXh0KSAqL1xuICBkZXNjcmlwdGlvbj86IHN0cmluZztcbiAgLyoqIGkxOG4ga2V5IGZvciBkZXNjcmlwdGlvbiAqL1xuICBkZXNjcmlwdGlvbktleT86IHN0cmluZztcbiAgLyoqIGkxOG4gbmFtZXNwYWNlIGZvciB0cmFuc2xhdGlvbnMgKi9cbiAgaTE4bk5hbWVzcGFjZT86IHN0cmluZztcblxuICAvKiogQ2FyZCBzaXplIHZhcmlhbnQgKi9cbiAgc2l6ZT86IEFjdGlvbkNhcmRTaXplO1xuICAvKiogU2hvdyBib3JkZXIgKi9cbiAgYm9yZGVyZWQ/OiBib29sZWFuO1xuICAvKiogQm9yZGVyIGNvbG9yICovXG4gIGJvcmRlckNvbG9yPzogQ29sb3IgfCBzdHJpbmc7XG4gIC8qKiBDYXJkIGJhY2tncm91bmQgY29sb3IgKi9cbiAgYmFja2dyb3VuZENvbG9yPzogQ29sb3IgfCBzdHJpbmc7XG4gIC8qKiBTaG93IHNoYWRvdyAqL1xuICBzaGFkb3dlZD86IGJvb2xlYW47XG5cbiAgLyoqIERpc2FibGVkIHN0YXRlICovXG4gIGRpc2FibGVkPzogYm9vbGVhbjtcbiAgLyoqIFJvdXRlciBsaW5rIGZvciBuYXZpZ2F0aW9uICovXG4gIHJvdXRlckxpbms/OiBzdHJpbmcgfCBhbnlbXTtcbiAgLyoqIEV4dGVybmFsIFVSTCAob3BlbnMgaW4gbmV3IHRhYi9icm93c2VyKSAqL1xuICBocmVmPzogc3RyaW5nO1xuXG4gIC8qKiBCYWRnZSBjb25maWd1cmF0aW9uICovXG4gIGJhZGdlPzogQWN0aW9uQ2FyZEJhZGdlO1xuICAvKiogU2hvdyBjaGV2cm9uIGljb24gb24gdGhlIHJpZ2h0ICovXG4gIHNob3dDaGV2cm9uPzogYm9vbGVhbjtcbn1cblxuLyoqXG4gKiBEZWZhdWx0IHZhbHVlcyBmb3IgQWN0aW9uQ2FyZE1ldGFkYXRhXG4gKi9cbmV4cG9ydCBjb25zdCBBQ1RJT05fQ0FSRF9ERUZBVUxUUzogUmVxdWlyZWQ8XG4gIFBpY2s8QWN0aW9uQ2FyZE1ldGFkYXRhLCAnc2l6ZScgfCAnYm9yZGVyZWQnIHwgJ3NoYWRvd2VkJyB8ICdkaXNhYmxlZCcgfCAnc2hvd0NoZXZyb24nPlxuPiA9IHtcbiAgc2l6ZTogJ21lZGl1bScsXG4gIGJvcmRlcmVkOiBmYWxzZSxcbiAgc2hhZG93ZWQ6IHRydWUsXG4gIGRpc2FibGVkOiBmYWxzZSxcbiAgc2hvd0NoZXZyb246IGZhbHNlLFxufTtcbiJdfQ==
@@ -0,0 +1,236 @@
1
+ import { Component, Input, computed } from '@angular/core';
2
+ import { CommonModule } from '@angular/common';
3
+ import { IonCard, IonCardHeader, IonCardTitle, IonCardContent, IonList, IonItem, IonLabel, IonIcon, IonButton, IonText, IonChip, AlertController, } from '@ionic/angular/standalone';
4
+ import { addIcons } from 'ionicons';
5
+ import { logoGoogle, logoApple, logoMicrosoft, linkOutline, unlinkOutline, checkmarkCircle, } from 'ionicons/icons';
6
+ import { OAUTH_PROVIDERS_INFO } from './types';
7
+ import * as i0 from "@angular/core";
8
+ import * as i1 from "@angular/common";
9
+ addIcons({
10
+ logoGoogle,
11
+ logoApple,
12
+ logoMicrosoft,
13
+ linkOutline,
14
+ unlinkOutline,
15
+ checkmarkCircle,
16
+ });
17
+ /**
18
+ * Linked Providers Component
19
+ *
20
+ * Muestra los proveedores OAuth vinculados al usuario y permite
21
+ * vincular nuevos o desvincular existentes.
22
+ *
23
+ * @example
24
+ * <val-linked-providers
25
+ * [props]="{
26
+ * providers: linkedProviders(),
27
+ * onLink: linkProvider,
28
+ * onUnlink: unlinkProvider
29
+ * }"
30
+ * />
31
+ */
32
+ export class LinkedProvidersComponent {
33
+ constructor() {
34
+ this.alertCtrl = new AlertController();
35
+ // Computed signals
36
+ this.linkedProviders = computed(() => this.props?.providers || []);
37
+ this.unlinkedProviders = computed(() => {
38
+ const linked = new Set(this.linkedProviders().map(p => p.provider));
39
+ const available = this.props?.availableProviders || ['google'];
40
+ return available
41
+ .filter(p => !linked.has(p))
42
+ .map(p => OAUTH_PROVIDERS_INFO[p]);
43
+ });
44
+ this.canUnlink = computed(() => {
45
+ // Can unlink if there's more than one provider or user has password
46
+ return this.linkedProviders().length > 1;
47
+ });
48
+ }
49
+ getProviderInfo(provider) {
50
+ return OAUTH_PROVIDERS_INFO[provider] || OAUTH_PROVIDERS_INFO.google;
51
+ }
52
+ onLinkProvider(provider) {
53
+ this.props.onLink?.(provider);
54
+ }
55
+ async confirmUnlink(provider) {
56
+ const info = this.getProviderInfo(provider);
57
+ const alert = await this.alertCtrl.create({
58
+ header: 'Desvincular cuenta',
59
+ message: `¿Estás seguro de que quieres desvincular tu cuenta de ${info.name}?`,
60
+ buttons: [
61
+ {
62
+ text: 'Cancelar',
63
+ role: 'cancel',
64
+ },
65
+ {
66
+ text: 'Desvincular',
67
+ role: 'destructive',
68
+ handler: () => {
69
+ this.props.onUnlink?.(provider);
70
+ },
71
+ },
72
+ ],
73
+ });
74
+ await alert.present();
75
+ }
76
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: LinkedProvidersComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
77
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: LinkedProvidersComponent, isStandalone: true, selector: "val-linked-providers", inputs: { props: "props" }, ngImport: i0, template: `
78
+ @if (props.compact) {
79
+ <ng-container *ngTemplateOutlet="providersList" />
80
+ } @else {
81
+ <ion-card>
82
+ <ion-card-header>
83
+ <ion-card-title>{{ props.title || 'Cuentas vinculadas' }}</ion-card-title>
84
+ </ion-card-header>
85
+ <ion-card-content>
86
+ @if (props.description) {
87
+ <p class="section-description">{{ props.description }}</p>
88
+ }
89
+ <ng-container *ngTemplateOutlet="providersList" />
90
+ </ion-card-content>
91
+ </ion-card>
92
+ }
93
+
94
+ <ng-template #providersList>
95
+ <ion-list lines="none" class="providers-list">
96
+ <!-- Linked providers -->
97
+ @for (provider of linkedProviders(); track provider.provider) {
98
+ <ion-item class="provider-item linked">
99
+ <div class="provider-icon" [style.background-color]="getProviderInfo(provider.provider).bgColor" slot="start">
100
+ <ion-icon [name]="getProviderInfo(provider.provider).icon" [style.color]="getProviderInfo(provider.provider).color" />
101
+ </div>
102
+ <ion-label>
103
+ <h3>{{ getProviderInfo(provider.provider).name }}</h3>
104
+ <p>{{ provider.email }}</p>
105
+ </ion-label>
106
+ <ion-icon name="checkmark-circle" color="success" slot="end" class="linked-icon" />
107
+ @if (props.allowUnlink !== false && canUnlink()) {
108
+ <ion-button
109
+ fill="clear"
110
+ color="medium"
111
+ slot="end"
112
+ (click)="confirmUnlink(provider.provider)"
113
+ class="unlink-btn"
114
+ >
115
+ <ion-icon name="unlink-outline" slot="icon-only" />
116
+ </ion-button>
117
+ }
118
+ </ion-item>
119
+ }
120
+
121
+ <!-- Available providers to link -->
122
+ @if (props.showLinkButton !== false) {
123
+ @for (provider of unlinkedProviders(); track provider.id) {
124
+ <ion-item class="provider-item available" button (click)="onLinkProvider(provider.id)">
125
+ <div class="provider-icon muted" slot="start">
126
+ <ion-icon [name]="provider.icon" />
127
+ </div>
128
+ <ion-label>
129
+ <h3>{{ provider.name }}</h3>
130
+ <p>Vincular cuenta</p>
131
+ </ion-label>
132
+ <ion-icon name="link-outline" color="primary" slot="end" />
133
+ </ion-item>
134
+ }
135
+ }
136
+
137
+ @if (linkedProviders().length === 0 && unlinkedProviders().length === 0) {
138
+ <ion-item>
139
+ <ion-label class="ion-text-center">
140
+ <p>No hay proveedores disponibles</p>
141
+ </ion-label>
142
+ </ion-item>
143
+ }
144
+ </ion-list>
145
+ </ng-template>
146
+ `, isInline: true, styles: ["ion-card{margin:0;border-radius:12px;box-shadow:0 1px 3px #00000014}ion-card-title{font-size:1.125rem;font-weight:600}.section-description{margin:0 0 1rem;color:var(--ion-color-medium);font-size:.875rem}.providers-list{padding:0}.provider-item{--padding-start: 0;--padding-end: 0;--inner-padding-end: 0;margin-bottom:.5rem;border-radius:8px;border:1px solid var(--ion-border-color, #e0e0e0)}.provider-item:last-child{margin-bottom:0}.provider-icon{display:flex;align-items:center;justify-content:center;width:40px;height:40px;border-radius:8px;margin-right:.75rem}.provider-icon ion-icon{font-size:1.25rem}.provider-icon.muted{background:var(--ion-color-light)}.provider-icon.muted ion-icon{color:var(--ion-color-medium)}.provider-item ion-label h3{font-weight:600;font-size:.9375rem;margin-bottom:.125rem}.provider-item ion-label p{font-size:.8125rem;color:var(--ion-color-medium)}.linked-icon{font-size:1.25rem}.unlink-btn{--padding-start: .5rem;--padding-end: .5rem}.provider-item.available{cursor:pointer}.provider-item.available:hover{background:var(--ion-color-light-tint)}:host-context(body.dark){.provider-item{border-color:var(--ion-color-step-150)}.provider-icon.muted{background:var(--ion-color-step-100)}}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "component", type: IonCard, selector: "ion-card", inputs: ["button", "color", "disabled", "download", "href", "mode", "rel", "routerAnimation", "routerDirection", "target", "type"] }, { kind: "component", type: IonCardHeader, selector: "ion-card-header", inputs: ["color", "mode", "translucent"] }, { kind: "component", type: IonCardTitle, selector: "ion-card-title", inputs: ["color", "mode"] }, { kind: "component", type: IonCardContent, selector: "ion-card-content", inputs: ["mode"] }, { kind: "component", type: IonList, selector: "ion-list", inputs: ["inset", "lines", "mode"] }, { kind: "component", type: IonItem, selector: "ion-item", inputs: ["button", "color", "detail", "detailIcon", "disabled", "download", "href", "lines", "mode", "rel", "routerAnimation", "routerDirection", "target", "type"] }, { kind: "component", type: IonLabel, selector: "ion-label", inputs: ["color", "mode", "position"] }, { kind: "component", type: IonIcon, selector: "ion-icon", inputs: ["color", "flipRtl", "icon", "ios", "lazy", "md", "mode", "name", "sanitize", "size", "src"] }, { kind: "component", type: IonButton, selector: "ion-button", inputs: ["buttonType", "color", "disabled", "download", "expand", "fill", "form", "href", "mode", "rel", "routerAnimation", "routerDirection", "shape", "size", "strong", "target", "type"] }] }); }
147
+ }
148
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: LinkedProvidersComponent, decorators: [{
149
+ type: Component,
150
+ args: [{ selector: 'val-linked-providers', standalone: true, imports: [
151
+ CommonModule,
152
+ IonCard,
153
+ IonCardHeader,
154
+ IonCardTitle,
155
+ IonCardContent,
156
+ IonList,
157
+ IonItem,
158
+ IonLabel,
159
+ IonIcon,
160
+ IonButton,
161
+ IonText,
162
+ IonChip,
163
+ ], template: `
164
+ @if (props.compact) {
165
+ <ng-container *ngTemplateOutlet="providersList" />
166
+ } @else {
167
+ <ion-card>
168
+ <ion-card-header>
169
+ <ion-card-title>{{ props.title || 'Cuentas vinculadas' }}</ion-card-title>
170
+ </ion-card-header>
171
+ <ion-card-content>
172
+ @if (props.description) {
173
+ <p class="section-description">{{ props.description }}</p>
174
+ }
175
+ <ng-container *ngTemplateOutlet="providersList" />
176
+ </ion-card-content>
177
+ </ion-card>
178
+ }
179
+
180
+ <ng-template #providersList>
181
+ <ion-list lines="none" class="providers-list">
182
+ <!-- Linked providers -->
183
+ @for (provider of linkedProviders(); track provider.provider) {
184
+ <ion-item class="provider-item linked">
185
+ <div class="provider-icon" [style.background-color]="getProviderInfo(provider.provider).bgColor" slot="start">
186
+ <ion-icon [name]="getProviderInfo(provider.provider).icon" [style.color]="getProviderInfo(provider.provider).color" />
187
+ </div>
188
+ <ion-label>
189
+ <h3>{{ getProviderInfo(provider.provider).name }}</h3>
190
+ <p>{{ provider.email }}</p>
191
+ </ion-label>
192
+ <ion-icon name="checkmark-circle" color="success" slot="end" class="linked-icon" />
193
+ @if (props.allowUnlink !== false && canUnlink()) {
194
+ <ion-button
195
+ fill="clear"
196
+ color="medium"
197
+ slot="end"
198
+ (click)="confirmUnlink(provider.provider)"
199
+ class="unlink-btn"
200
+ >
201
+ <ion-icon name="unlink-outline" slot="icon-only" />
202
+ </ion-button>
203
+ }
204
+ </ion-item>
205
+ }
206
+
207
+ <!-- Available providers to link -->
208
+ @if (props.showLinkButton !== false) {
209
+ @for (provider of unlinkedProviders(); track provider.id) {
210
+ <ion-item class="provider-item available" button (click)="onLinkProvider(provider.id)">
211
+ <div class="provider-icon muted" slot="start">
212
+ <ion-icon [name]="provider.icon" />
213
+ </div>
214
+ <ion-label>
215
+ <h3>{{ provider.name }}</h3>
216
+ <p>Vincular cuenta</p>
217
+ </ion-label>
218
+ <ion-icon name="link-outline" color="primary" slot="end" />
219
+ </ion-item>
220
+ }
221
+ }
222
+
223
+ @if (linkedProviders().length === 0 && unlinkedProviders().length === 0) {
224
+ <ion-item>
225
+ <ion-label class="ion-text-center">
226
+ <p>No hay proveedores disponibles</p>
227
+ </ion-label>
228
+ </ion-item>
229
+ }
230
+ </ion-list>
231
+ </ng-template>
232
+ `, styles: ["ion-card{margin:0;border-radius:12px;box-shadow:0 1px 3px #00000014}ion-card-title{font-size:1.125rem;font-weight:600}.section-description{margin:0 0 1rem;color:var(--ion-color-medium);font-size:.875rem}.providers-list{padding:0}.provider-item{--padding-start: 0;--padding-end: 0;--inner-padding-end: 0;margin-bottom:.5rem;border-radius:8px;border:1px solid var(--ion-border-color, #e0e0e0)}.provider-item:last-child{margin-bottom:0}.provider-icon{display:flex;align-items:center;justify-content:center;width:40px;height:40px;border-radius:8px;margin-right:.75rem}.provider-icon ion-icon{font-size:1.25rem}.provider-icon.muted{background:var(--ion-color-light)}.provider-icon.muted ion-icon{color:var(--ion-color-medium)}.provider-item ion-label h3{font-weight:600;font-size:.9375rem;margin-bottom:.125rem}.provider-item ion-label p{font-size:.8125rem;color:var(--ion-color-medium)}.linked-icon{font-size:1.25rem}.unlink-btn{--padding-start: .5rem;--padding-end: .5rem}.provider-item.available{cursor:pointer}.provider-item.available:hover{background:var(--ion-color-light-tint)}:host-context(body.dark){.provider-item{border-color:var(--ion-color-step-150)}.provider-icon.muted{background:var(--ion-color-step-100)}}\n"] }]
233
+ }], propDecorators: { props: [{
234
+ type: Input
235
+ }] } });
236
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"linked-providers.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/molecules/linked-providers/linked-providers.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, computed, signal } from '@angular/core';\nimport { CommonModule } from '@angular/common';\nimport {\n  IonCard,\n  IonCardHeader,\n  IonCardTitle,\n  IonCardContent,\n  IonList,\n  IonItem,\n  IonLabel,\n  IonIcon,\n  IonButton,\n  IonText,\n  IonChip,\n  AlertController,\n} from '@ionic/angular/standalone';\nimport { addIcons } from 'ionicons';\nimport {\n  logoGoogle,\n  logoApple,\n  logoMicrosoft,\n  linkOutline,\n  unlinkOutline,\n  checkmarkCircle,\n} from 'ionicons/icons';\nimport { LinkedProvidersMetadata, OAUTH_PROVIDERS_INFO, ProviderDisplayInfo } from './types';\nimport { LinkedProvider, OAuthProvider } from '../../../services/auth/types';\n\naddIcons({\n  logoGoogle,\n  logoApple,\n  logoMicrosoft,\n  linkOutline,\n  unlinkOutline,\n  checkmarkCircle,\n});\n\n/**\n * Linked Providers Component\n *\n * Muestra los proveedores OAuth vinculados al usuario y permite\n * vincular nuevos o desvincular existentes.\n *\n * @example\n * <val-linked-providers\n *   [props]=\"{\n *     providers: linkedProviders(),\n *     onLink: linkProvider,\n *     onUnlink: unlinkProvider\n *   }\"\n * />\n */\n@Component({\n  selector: 'val-linked-providers',\n  standalone: true,\n  imports: [\n    CommonModule,\n    IonCard,\n    IonCardHeader,\n    IonCardTitle,\n    IonCardContent,\n    IonList,\n    IonItem,\n    IonLabel,\n    IonIcon,\n    IonButton,\n    IonText,\n    IonChip,\n  ],\n  template: `\n    @if (props.compact) {\n      <ng-container *ngTemplateOutlet=\"providersList\" />\n    } @else {\n      <ion-card>\n        <ion-card-header>\n          <ion-card-title>{{ props.title || 'Cuentas vinculadas' }}</ion-card-title>\n        </ion-card-header>\n        <ion-card-content>\n          @if (props.description) {\n            <p class=\"section-description\">{{ props.description }}</p>\n          }\n          <ng-container *ngTemplateOutlet=\"providersList\" />\n        </ion-card-content>\n      </ion-card>\n    }\n\n    <ng-template #providersList>\n      <ion-list lines=\"none\" class=\"providers-list\">\n        <!-- Linked providers -->\n        @for (provider of linkedProviders(); track provider.provider) {\n          <ion-item class=\"provider-item linked\">\n            <div class=\"provider-icon\" [style.background-color]=\"getProviderInfo(provider.provider).bgColor\" slot=\"start\">\n              <ion-icon [name]=\"getProviderInfo(provider.provider).icon\" [style.color]=\"getProviderInfo(provider.provider).color\" />\n            </div>\n            <ion-label>\n              <h3>{{ getProviderInfo(provider.provider).name }}</h3>\n              <p>{{ provider.email }}</p>\n            </ion-label>\n            <ion-icon name=\"checkmark-circle\" color=\"success\" slot=\"end\" class=\"linked-icon\" />\n            @if (props.allowUnlink !== false && canUnlink()) {\n              <ion-button\n                fill=\"clear\"\n                color=\"medium\"\n                slot=\"end\"\n                (click)=\"confirmUnlink(provider.provider)\"\n                class=\"unlink-btn\"\n              >\n                <ion-icon name=\"unlink-outline\" slot=\"icon-only\" />\n              </ion-button>\n            }\n          </ion-item>\n        }\n\n        <!-- Available providers to link -->\n        @if (props.showLinkButton !== false) {\n          @for (provider of unlinkedProviders(); track provider.id) {\n            <ion-item class=\"provider-item available\" button (click)=\"onLinkProvider(provider.id)\">\n              <div class=\"provider-icon muted\" slot=\"start\">\n                <ion-icon [name]=\"provider.icon\" />\n              </div>\n              <ion-label>\n                <h3>{{ provider.name }}</h3>\n                <p>Vincular cuenta</p>\n              </ion-label>\n              <ion-icon name=\"link-outline\" color=\"primary\" slot=\"end\" />\n            </ion-item>\n          }\n        }\n\n        @if (linkedProviders().length === 0 && unlinkedProviders().length === 0) {\n          <ion-item>\n            <ion-label class=\"ion-text-center\">\n              <p>No hay proveedores disponibles</p>\n            </ion-label>\n          </ion-item>\n        }\n      </ion-list>\n    </ng-template>\n  `,\n  styles: [`\n    ion-card {\n      margin: 0;\n      border-radius: 12px;\n      box-shadow: 0 1px 3px rgba(0, 0, 0, 0.08);\n    }\n\n    ion-card-title {\n      font-size: 1.125rem;\n      font-weight: 600;\n    }\n\n    .section-description {\n      margin: 0 0 1rem;\n      color: var(--ion-color-medium);\n      font-size: 0.875rem;\n    }\n\n    .providers-list {\n      padding: 0;\n    }\n\n    .provider-item {\n      --padding-start: 0;\n      --padding-end: 0;\n      --inner-padding-end: 0;\n      margin-bottom: 0.5rem;\n      border-radius: 8px;\n      border: 1px solid var(--ion-border-color, #e0e0e0);\n    }\n\n    .provider-item:last-child {\n      margin-bottom: 0;\n    }\n\n    .provider-icon {\n      display: flex;\n      align-items: center;\n      justify-content: center;\n      width: 40px;\n      height: 40px;\n      border-radius: 8px;\n      margin-right: 0.75rem;\n    }\n\n    .provider-icon ion-icon {\n      font-size: 1.25rem;\n    }\n\n    .provider-icon.muted {\n      background: var(--ion-color-light);\n    }\n\n    .provider-icon.muted ion-icon {\n      color: var(--ion-color-medium);\n    }\n\n    .provider-item ion-label h3 {\n      font-weight: 600;\n      font-size: 0.9375rem;\n      margin-bottom: 0.125rem;\n    }\n\n    .provider-item ion-label p {\n      font-size: 0.8125rem;\n      color: var(--ion-color-medium);\n    }\n\n    .linked-icon {\n      font-size: 1.25rem;\n    }\n\n    .unlink-btn {\n      --padding-start: 0.5rem;\n      --padding-end: 0.5rem;\n    }\n\n    .provider-item.available {\n      cursor: pointer;\n    }\n\n    .provider-item.available:hover {\n      background: var(--ion-color-light-tint);\n    }\n\n    /* Dark mode */\n    :host-context(body.dark) {\n      .provider-item {\n        border-color: var(--ion-color-step-150);\n      }\n\n      .provider-icon.muted {\n        background: var(--ion-color-step-100);\n      }\n    }\n  `],\n})\nexport class LinkedProvidersComponent {\n  @Input() props!: LinkedProvidersMetadata;\n\n  private alertCtrl = new AlertController();\n\n  // Computed signals\n  linkedProviders = computed(() => this.props?.providers || []);\n\n  unlinkedProviders = computed(() => {\n    const linked = new Set(this.linkedProviders().map(p => p.provider));\n    const available = this.props?.availableProviders || ['google'];\n    return available\n      .filter(p => !linked.has(p))\n      .map(p => OAUTH_PROVIDERS_INFO[p]);\n  });\n\n  canUnlink = computed(() => {\n    // Can unlink if there's more than one provider or user has password\n    return this.linkedProviders().length > 1;\n  });\n\n  getProviderInfo(provider: OAuthProvider): ProviderDisplayInfo {\n    return OAUTH_PROVIDERS_INFO[provider] || OAUTH_PROVIDERS_INFO.google;\n  }\n\n  onLinkProvider(provider: OAuthProvider): void {\n    this.props.onLink?.(provider);\n  }\n\n  async confirmUnlink(provider: OAuthProvider): Promise<void> {\n    const info = this.getProviderInfo(provider);\n\n    const alert = await this.alertCtrl.create({\n      header: 'Desvincular cuenta',\n      message: `¿Estás seguro de que quieres desvincular tu cuenta de ${info.name}?`,\n      buttons: [\n        {\n          text: 'Cancelar',\n          role: 'cancel',\n        },\n        {\n          text: 'Desvincular',\n          role: 'destructive',\n          handler: () => {\n            this.props.onUnlink?.(provider);\n          },\n        },\n      ],\n    });\n\n    await alert.present();\n  }\n}\n"]}