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.
- 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/username-input/types.mjs +2 -0
- package/esm2022/lib/components/molecules/username-input/username-input.component.mjs +260 -0
- 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 +870 -42
- 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/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,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"]}
|