valtech-components 2.0.659 → 2.0.660

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,15 @@
1
+ /**
2
+ * Default values
3
+ */
4
+ export const WELCOME_CARD_DEFAULTS = {
5
+ variant: 'gradient',
6
+ gradientAngle: 135,
7
+ showParticles: true,
8
+ loading: false,
9
+ avatar: {
10
+ mode: 'initials',
11
+ size: 'large',
12
+ showRing: true,
13
+ },
14
+ };
15
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL2NvbXBvbmVudHMvbW9sZWN1bGVzL3dlbGNvbWUtY2FyZC90eXBlcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUErSEE7O0dBRUc7QUFDSCxNQUFNLENBQUMsTUFBTSxxQkFBcUIsR0FJOUI7SUFDRixPQUFPLEVBQUUsVUFBVTtJQUNuQixhQUFhLEVBQUUsR0FBRztJQUNsQixhQUFhLEVBQUUsSUFBSTtJQUNuQixPQUFPLEVBQUUsS0FBSztJQUNkLE1BQU0sRUFBRTtRQUNOLElBQUksRUFBRSxVQUFVO1FBQ2hCLElBQUksRUFBRSxPQUFPO1FBQ2IsUUFBUSxFQUFFLElBQUk7S0FDZjtDQUNGLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDb2xvciB9IGZyb20gJ0Bpb25pYy9jb3JlJztcblxuLyoqXG4gKiBBdmF0YXIgbW9kZSBmb3Igd2VsY29tZS1jYXJkXG4gKi9cbmV4cG9ydCB0eXBlIFdlbGNvbWVBdmF0YXJNb2RlID0gJ2luaXRpYWxzJyB8ICdpbWFnZScgfCAnaWNvbic7XG5cbi8qKlxuICogVmlzdWFsIHZhcmlhbnQgZm9yIHRoZSBjYXJkXG4gKi9cbmV4cG9ydCB0eXBlIFdlbGNvbWVDYXJkVmFyaWFudCA9ICdkZWZhdWx0JyB8ICdncmFkaWVudCcgfCAnZ2xhc3MnIHwgJ21pbmltYWwnO1xuXG4vKipcbiAqIFNpemUgb3B0aW9ucyBmb3IgYXZhdGFyXG4gKi9cbmV4cG9ydCB0eXBlIFdlbGNvbWVBdmF0YXJTaXplID0gJ3NtYWxsJyB8ICdtZWRpdW0nIHwgJ2xhcmdlJztcblxuLyoqXG4gKiBBdmF0YXIgY29uZmlndXJhdGlvblxuICovXG5leHBvcnQgaW50ZXJmYWNlIFdlbGNvbWVBdmF0YXJDb25maWcge1xuICAvKiogQXZhdGFyIGRpc3BsYXkgbW9kZSAqL1xuICBtb2RlPzogV2VsY29tZUF2YXRhck1vZGU7XG4gIC8qKiBJbWFnZSBVUkwgKGZvciBtb2RlOiAnaW1hZ2UnKSAqL1xuICBpbWFnZVVybD86IHN0cmluZztcbiAgLyoqIElvbmljb24gbmFtZSAoZm9yIG1vZGU6ICdpY29uJykgKi9cbiAgaWNvbj86IHN0cmluZztcbiAgLyoqIEN1c3RvbSBpbml0aWFscyAob3ZlcnJpZGVzIGF1dG8tZ2VuZXJhdGVkIGZyb20gbmFtZSkgKi9cbiAgaW5pdGlhbHM/OiBzdHJpbmc7XG4gIC8qKiBBdmF0YXIgc2l6ZSAqL1xuICBzaXplPzogV2VsY29tZUF2YXRhclNpemU7XG4gIC8qKiBTaG93IGFuaW1hdGVkIHJpbmcgYXJvdW5kIGF2YXRhciAqL1xuICBzaG93UmluZz86IGJvb2xlYW47XG4gIC8qKiBSaW5nIGNvbG9yICovXG4gIHJpbmdDb2xvcj86IENvbG9yIHwgc3RyaW5nO1xufVxuXG4vKipcbiAqIFVzZXIgZGF0YSBmb3IgdGhlIHdlbGNvbWUgY2FyZFxuICovXG5leHBvcnQgaW50ZXJmYWNlIFdlbGNvbWVVc2VyRGF0YSB7XG4gIC8qKiBVc2VyJ3MgZGlzcGxheSBuYW1lICovXG4gIG5hbWU/OiBzdHJpbmc7XG4gIC8qKiBVc2VyJ3MgZW1haWwgKi9cbiAgZW1haWw/OiBzdHJpbmc7XG4gIC8qKiBVc2VyJ3MgYXZhdGFyIFVSTCAob3B0aW9uYWwpICovXG4gIGF2YXRhclVybD86IHN0cmluZztcbiAgLyoqIEN1c3RvbSBpbml0aWFscyAob3ZlcnJpZGVzIGF1dG8tZ2VuZXJhdGVkKSAqL1xuICBpbml0aWFscz86IHN0cmluZztcbn1cblxuLyoqXG4gKiBBY3Rpb24gYnV0dG9uIGNvbmZpZ3VyYXRpb25cbiAqL1xuZXhwb3J0IGludGVyZmFjZSBXZWxjb21lQ2FyZEFjdGlvbiB7XG4gIC8qKiBCdXR0b24gbGFiZWwgKi9cbiAgbGFiZWw6IHN0cmluZztcbiAgLyoqIGkxOG4ga2V5IGZvciBsYWJlbCAqL1xuICBsYWJlbEtleT86IHN0cmluZztcbiAgLyoqIElvbmljb24gbmFtZSAqL1xuICBpY29uPzogc3RyaW5nO1xuICAvKiogUm91dGVyIGxpbmsgKi9cbiAgcm91dGVyTGluaz86IHN0cmluZyB8IGFueVtdO1xuICAvKiogQWN0aW9uIGlkZW50aWZpZXIgKi9cbiAgdG9rZW4/OiBzdHJpbmc7XG4gIC8qKiBCdXR0b24gY29sb3IgKi9cbiAgY29sb3I/OiBDb2xvciB8IHN0cmluZztcbn1cblxuLyoqXG4gKiBDbGljayBldmVudCBlbWl0dGVkIGJ5IHdlbGNvbWUtY2FyZFxuICovXG5leHBvcnQgaW50ZXJmYWNlIFdlbGNvbWVDYXJkQ2xpY2tFdmVudCB7XG4gIC8qKiBDbGljayB0YXJnZXQgKi9cbiAgdGFyZ2V0OiAnY2FyZCcgfCAnYXZhdGFyJyB8ICdhY3Rpb24nO1xuICAvKiogQWN0aW9uIHRva2VuIGlmIGFjdGlvbiB3YXMgY2xpY2tlZCAqL1xuICBhY3Rpb25Ub2tlbj86IHN0cmluZztcbn1cblxuLyoqXG4gKiBNZXRhZGF0YSBmb3IgdmFsLXdlbGNvbWUtY2FyZCBjb21wb25lbnRcbiAqL1xuZXhwb3J0IGludGVyZmFjZSBXZWxjb21lQ2FyZE1ldGFkYXRhIHtcbiAgLyoqIENhcmQgdmlzdWFsIHZhcmlhbnQgKi9cbiAgdmFyaWFudD86IFdlbGNvbWVDYXJkVmFyaWFudDtcblxuICAvKiogVXNlciBkYXRhICovXG4gIHVzZXI/OiBXZWxjb21lVXNlckRhdGE7XG5cbiAgLyoqIEdyZWV0aW5nIHRleHQgKGUuZy4sIFwiV2VsY29tZSBiYWNrXCIpICovXG4gIGdyZWV0aW5nPzogc3RyaW5nO1xuICAvKiogaTE4biBrZXkgZm9yIGdyZWV0aW5nICovXG4gIGdyZWV0aW5nS2V5Pzogc3RyaW5nO1xuICAvKiogaTE4biBuYW1lc3BhY2UgKi9cbiAgaTE4bk5hbWVzcGFjZT86IHN0cmluZztcblxuICAvKiogU3VidGl0bGUgdGV4dCAoc2hvd24gYmVsb3cgbmFtZSkgKi9cbiAgc3VidGl0bGU/OiBzdHJpbmc7XG4gIC8qKiBpMThuIGtleSBmb3Igc3VidGl0bGUgKi9cbiAgc3VidGl0bGVLZXk/OiBzdHJpbmc7XG5cbiAgLyoqIEF2YXRhciBjb25maWd1cmF0aW9uICovXG4gIGF2YXRhcj86IFdlbGNvbWVBdmF0YXJDb25maWc7XG5cbiAgLyoqIFByaW1hcnkgZ3JhZGllbnQgY29sb3IgKi9cbiAgZ3JhZGllbnRGcm9tPzogQ29sb3IgfCBzdHJpbmc7XG4gIC8qKiBTZWNvbmRhcnkgZ3JhZGllbnQgY29sb3IgKi9cbiAgZ3JhZGllbnRUbz86IENvbG9yIHwgc3RyaW5nO1xuICAvKiogR3JhZGllbnQgZGlyZWN0aW9uIChkZWdyZWVzKSAqL1xuICBncmFkaWVudEFuZ2xlPzogbnVtYmVyO1xuXG4gIC8qKiBCYWNrZ3JvdW5kIGNvbG9yIChmb3Igbm9uLWdyYWRpZW50IHZhcmlhbnRzKSAqL1xuICBiYWNrZ3JvdW5kQ29sb3I/OiBDb2xvciB8IHN0cmluZztcblxuICAvKiogU2hvdyBhbmltYXRlZCBiYWNrZ3JvdW5kIHBhcnRpY2xlcyAqL1xuICBzaG93UGFydGljbGVzPzogYm9vbGVhbjtcblxuICAvKiogQWN0aW9uIGJ1dHRvbnMgKi9cbiAgYWN0aW9ucz86IFdlbGNvbWVDYXJkQWN0aW9uW107XG5cbiAgLyoqIFNob3cgc2tlbGV0b24gbG9hZGluZyBzdGF0ZSAqL1xuICBsb2FkaW5nPzogYm9vbGVhbjtcblxuICAvKiogVW5pcXVlIHRva2VuIGZvciBpZGVudGlmaWNhdGlvbiAqL1xuICB0b2tlbj86IHN0cmluZztcbn1cblxuLyoqXG4gKiBEZWZhdWx0IHZhbHVlc1xuICovXG5leHBvcnQgY29uc3QgV0VMQ09NRV9DQVJEX0RFRkFVTFRTOiBSZXF1aXJlZDxcbiAgUGljazxXZWxjb21lQ2FyZE1ldGFkYXRhLCAndmFyaWFudCcgfCAnZ3JhZGllbnRBbmdsZScgfCAnc2hvd1BhcnRpY2xlcycgfCAnbG9hZGluZyc+XG4+ICYge1xuICBhdmF0YXI6IFJlcXVpcmVkPFBpY2s8V2VsY29tZUF2YXRhckNvbmZpZywgJ21vZGUnIHwgJ3NpemUnIHwgJ3Nob3dSaW5nJz4+O1xufSA9IHtcbiAgdmFyaWFudDogJ2dyYWRpZW50JyxcbiAgZ3JhZGllbnRBbmdsZTogMTM1LFxuICBzaG93UGFydGljbGVzOiB0cnVlLFxuICBsb2FkaW5nOiBmYWxzZSxcbiAgYXZhdGFyOiB7XG4gICAgbW9kZTogJ2luaXRpYWxzJyxcbiAgICBzaXplOiAnbGFyZ2UnLFxuICAgIHNob3dSaW5nOiB0cnVlLFxuICB9LFxufTtcbiJdfQ==
@@ -0,0 +1,394 @@
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, IonSkeletonText } from '@ionic/angular/standalone';
5
+ import { addIcons } from 'ionicons';
6
+ import { personOutline, chevronForwardOutline } from 'ionicons/icons';
7
+ import { I18nService } from '../../../services/i18n';
8
+ import { ThemeService } from '../../../services/theme.service';
9
+ import { WELCOME_CARD_DEFAULTS, } from './types';
10
+ import * as i0 from "@angular/core";
11
+ addIcons({ personOutline, chevronForwardOutline });
12
+ const IONIC_COLORS = [
13
+ 'primary', 'secondary', 'tertiary', 'success',
14
+ 'warning', 'danger', 'light', 'medium', 'dark'
15
+ ];
16
+ // Vibrant color palette for avatar backgrounds
17
+ const AVATAR_COLORS = [
18
+ { bg: '#6366f1', text: '#ffffff' }, // Indigo
19
+ { bg: '#8b5cf6', text: '#ffffff' }, // Violet
20
+ { bg: '#ec4899', text: '#ffffff' }, // Pink
21
+ { bg: '#f43f5e', text: '#ffffff' }, // Rose
22
+ { bg: '#f97316', text: '#ffffff' }, // Orange
23
+ { bg: '#eab308', text: '#1f2937' }, // Yellow
24
+ { bg: '#22c55e', text: '#ffffff' }, // Green
25
+ { bg: '#14b8a6', text: '#ffffff' }, // Teal
26
+ { bg: '#06b6d4', text: '#ffffff' }, // Cyan
27
+ { bg: '#3b82f6', text: '#ffffff' }, // Blue
28
+ ];
29
+ /**
30
+ * val-welcome-card
31
+ *
32
+ * A modern welcome/greeting card with avatar, animated gradient background,
33
+ * and optional action buttons. Perfect for dashboard headers.
34
+ *
35
+ * @example Basic usage
36
+ * ```html
37
+ * <val-welcome-card
38
+ * [props]="{
39
+ * greeting: 'Welcome back',
40
+ * user: { name: 'John Doe', email: 'john@example.com' }
41
+ * }"
42
+ * />
43
+ * ```
44
+ *
45
+ * @example With gradient variant
46
+ * ```html
47
+ * <val-welcome-card
48
+ * [props]="{
49
+ * variant: 'gradient',
50
+ * greeting: 'Good morning',
51
+ * user: { name: 'Jane', email: 'jane@example.com' },
52
+ * gradientFrom: 'primary',
53
+ * gradientTo: 'tertiary'
54
+ * }"
55
+ * />
56
+ * ```
57
+ *
58
+ * @example With i18n
59
+ * ```html
60
+ * <val-welcome-card
61
+ * [props]="{
62
+ * greetingKey: 'welcome',
63
+ * i18nNamespace: 'Dashboard',
64
+ * user: user()
65
+ * }"
66
+ * />
67
+ * ```
68
+ */
69
+ export class WelcomeCardComponent {
70
+ constructor() {
71
+ this.i18n = inject(I18nService);
72
+ this.themeService = inject(ThemeService);
73
+ /** Component configuration */
74
+ this.props = input({});
75
+ /** Event emitted when card elements are clicked */
76
+ this.cardClick = new EventEmitter();
77
+ /** Merged configuration with defaults */
78
+ this.config = computed(() => ({
79
+ ...WELCOME_CARD_DEFAULTS,
80
+ ...this.props(),
81
+ }));
82
+ /** Avatar configuration with defaults */
83
+ this.avatarConfig = computed(() => ({
84
+ ...WELCOME_CARD_DEFAULTS.avatar,
85
+ ...this.props().avatar,
86
+ }));
87
+ /** Current theme mode */
88
+ this.isDark = computed(() => this.themeService.IsDark);
89
+ /** Color index based on user name hash */
90
+ this.colorIndex = computed(() => {
91
+ const name = this.config().user?.name || '';
92
+ let hash = 0;
93
+ for (let i = 0; i < name.length; i++) {
94
+ hash = name.charCodeAt(i) + ((hash << 5) - hash);
95
+ }
96
+ return Math.abs(hash) % AVATAR_COLORS.length;
97
+ });
98
+ }
99
+ /** Get initials from user name */
100
+ getInitials() {
101
+ const props = this.props();
102
+ // Check for explicit initials first (from props, not merged config)
103
+ if (props.avatar?.initials || props.user?.initials) {
104
+ return props.avatar?.initials || props.user?.initials || '';
105
+ }
106
+ const name = props.user?.name || '';
107
+ if (!name)
108
+ return '?';
109
+ const parts = name.trim().split(/\s+/);
110
+ if (parts.length === 1) {
111
+ return parts[0].substring(0, 2).toUpperCase();
112
+ }
113
+ return (parts[0][0] + parts[parts.length - 1][0]).toUpperCase();
114
+ }
115
+ /** Get greeting text with i18n support */
116
+ getGreeting() {
117
+ const cfg = this.config();
118
+ if (cfg.i18nNamespace && cfg.greetingKey) {
119
+ this.i18n.lang(); // Track for reactivity
120
+ return this.i18n.t(cfg.greetingKey, cfg.i18nNamespace);
121
+ }
122
+ return cfg.greeting || '';
123
+ }
124
+ /** Get subtitle text with i18n support */
125
+ getSubtitle() {
126
+ const cfg = this.config();
127
+ if (cfg.i18nNamespace && cfg.subtitleKey) {
128
+ this.i18n.lang();
129
+ return this.i18n.t(cfg.subtitleKey, cfg.i18nNamespace);
130
+ }
131
+ return cfg.subtitle || '';
132
+ }
133
+ /** Get action label with i18n support */
134
+ getActionLabel(action) {
135
+ const cfg = this.config();
136
+ if (cfg.i18nNamespace && action.labelKey) {
137
+ this.i18n.lang();
138
+ return this.i18n.t(action.labelKey, cfg.i18nNamespace);
139
+ }
140
+ return action.label || '';
141
+ }
142
+ /** Resolve color value */
143
+ resolveColor(color) {
144
+ if (!color)
145
+ return null;
146
+ if (IONIC_COLORS.includes(color)) {
147
+ return `var(--ion-color-${color})`;
148
+ }
149
+ return color;
150
+ }
151
+ /** Get gradient start color */
152
+ getGradientFrom() {
153
+ const color = this.config().gradientFrom;
154
+ return this.resolveColor(color) || 'var(--ion-color-primary)';
155
+ }
156
+ /** Get gradient end color */
157
+ getGradientTo() {
158
+ const color = this.config().gradientTo;
159
+ return this.resolveColor(color) || 'var(--ion-color-tertiary)';
160
+ }
161
+ /** Get background color (for non-gradient variants) */
162
+ getBackgroundColor() {
163
+ return this.resolveColor(this.config().backgroundColor);
164
+ }
165
+ /** Get avatar background based on user name */
166
+ getAvatarBackground() {
167
+ const cfg = this.avatarConfig();
168
+ if (cfg.mode === 'image')
169
+ return 'transparent';
170
+ return AVATAR_COLORS[this.colorIndex()].bg;
171
+ }
172
+ /** Get avatar text color */
173
+ getAvatarTextColor() {
174
+ return AVATAR_COLORS[this.colorIndex()].text;
175
+ }
176
+ /** Get ring color */
177
+ getRingColor() {
178
+ const ringColor = this.avatarConfig().ringColor;
179
+ return this.resolveColor(ringColor) || 'rgba(255, 255, 255, 0.5)';
180
+ }
181
+ /** Handle card click */
182
+ onCardClick(event) {
183
+ // Don't emit if clicking avatar or actions
184
+ if (event.target.closest('.welcome-card__avatar, .welcome-card__action')) {
185
+ return;
186
+ }
187
+ this.cardClick.emit({ target: 'card' });
188
+ }
189
+ /** Handle avatar click */
190
+ onAvatarClick(event) {
191
+ event.stopPropagation();
192
+ this.cardClick.emit({ target: 'avatar' });
193
+ }
194
+ /** Handle action click */
195
+ onActionClick(event, token) {
196
+ event.stopPropagation();
197
+ this.cardClick.emit({ target: 'action', actionToken: token });
198
+ }
199
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: WelcomeCardComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
200
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: WelcomeCardComponent, isStandalone: true, selector: "val-welcome-card", inputs: { props: { classPropertyName: "props", publicName: "props", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { cardClick: "cardClick" }, ngImport: i0, template: `
201
+ <article
202
+ class="welcome-card"
203
+ [class.welcome-card--default]="config().variant === 'default'"
204
+ [class.welcome-card--gradient]="config().variant === 'gradient'"
205
+ [class.welcome-card--glass]="config().variant === 'glass'"
206
+ [class.welcome-card--minimal]="config().variant === 'minimal'"
207
+ [class.welcome-card--dark]="isDark()"
208
+ [class.welcome-card--loading]="config().loading"
209
+ [style.--gradient-from]="getGradientFrom()"
210
+ [style.--gradient-to]="getGradientTo()"
211
+ [style.--gradient-angle]="config().gradientAngle + 'deg'"
212
+ [style.--bg-color]="getBackgroundColor()"
213
+ (click)="onCardClick($event)"
214
+ >
215
+ <!-- Animated particles background -->
216
+ @if (config().showParticles && config().variant === 'gradient') {
217
+ <div class="welcome-card__particles">
218
+ @for (i of [1,2,3,4,5,6]; track i) {
219
+ <span class="particle" [class]="'particle--' + i"></span>
220
+ }
221
+ </div>
222
+ }
223
+
224
+ <!-- Content -->
225
+ <div class="welcome-card__content">
226
+ <!-- Avatar -->
227
+ <div
228
+ class="welcome-card__avatar"
229
+ [class.welcome-card__avatar--small]="avatarConfig().size === 'small'"
230
+ [class.welcome-card__avatar--medium]="avatarConfig().size === 'medium'"
231
+ [class.welcome-card__avatar--large]="avatarConfig().size === 'large'"
232
+ [class.welcome-card__avatar--ring]="avatarConfig().showRing"
233
+ [style.--avatar-bg]="getAvatarBackground()"
234
+ [style.--avatar-color]="getAvatarTextColor()"
235
+ [style.--ring-color]="getRingColor()"
236
+ (click)="onAvatarClick($event)"
237
+ >
238
+ @if (config().loading) {
239
+ <ion-skeleton-text [animated]="true" class="avatar-skeleton"></ion-skeleton-text>
240
+ } @else if (avatarConfig().mode === 'image' && (avatarConfig().imageUrl || config().user?.avatarUrl)) {
241
+ <img
242
+ [src]="avatarConfig().imageUrl || config().user?.avatarUrl"
243
+ [alt]="config().user?.name || 'Avatar'"
244
+ class="avatar-image"
245
+ />
246
+ } @else if (avatarConfig().mode === 'icon') {
247
+ <ion-icon [name]="avatarConfig().icon || 'person-outline'"></ion-icon>
248
+ } @else {
249
+ <span class="avatar-initials">{{ getInitials() }}</span>
250
+ }
251
+ </div>
252
+
253
+ <!-- Text content -->
254
+ <div class="welcome-card__text">
255
+ @if (config().loading) {
256
+ <ion-skeleton-text [animated]="true" style="width: 40%; height: 14px; margin-bottom: 8px;"></ion-skeleton-text>
257
+ <ion-skeleton-text [animated]="true" style="width: 60%; height: 24px; margin-bottom: 4px;"></ion-skeleton-text>
258
+ <ion-skeleton-text [animated]="true" style="width: 50%; height: 14px;"></ion-skeleton-text>
259
+ } @else {
260
+ @if (getGreeting()) {
261
+ <span class="welcome-card__greeting">{{ getGreeting() }}</span>
262
+ }
263
+ <h2 class="welcome-card__name">{{ config().user?.name || 'User' }}</h2>
264
+ @if (getSubtitle() || config().user?.email) {
265
+ <span class="welcome-card__subtitle">{{ getSubtitle() || config().user?.email }}</span>
266
+ }
267
+ }
268
+ </div>
269
+ </div>
270
+
271
+ <!-- Actions -->
272
+ @if (config().actions?.length && !config().loading) {
273
+ <div class="welcome-card__actions">
274
+ @for (action of config().actions; track action.token || action.label) {
275
+ <button
276
+ type="button"
277
+ class="welcome-card__action"
278
+ [routerLink]="action.routerLink"
279
+ (click)="onActionClick($event, action.token)"
280
+ >
281
+ @if (action.icon) {
282
+ <ion-icon [name]="action.icon"></ion-icon>
283
+ }
284
+ <span>{{ getActionLabel(action) }}</span>
285
+ <ion-icon name="chevron-forward-outline" class="action-chevron"></ion-icon>
286
+ </button>
287
+ }
288
+ </div>
289
+ }
290
+
291
+ <ion-ripple-effect></ion-ripple-effect>
292
+ </article>
293
+ `, isInline: true, styles: [":host{display:block;width:100%}.welcome-card{position:relative;border-radius:20px;padding:1.5rem;overflow:hidden;cursor:pointer;transition:transform .3s ease,box-shadow .3s ease}.welcome-card--default{background:var(--bg-color, var(--ion-card-background, #fff));box-shadow:0 4px 20px #00000014}.welcome-card--default .welcome-card__greeting,.welcome-card--default .welcome-card__subtitle{color:var(--ion-color-medium)}.welcome-card--default .welcome-card__name{color:var(--ion-text-color)}.welcome-card--gradient{background:linear-gradient(var(--gradient-angle, 135deg),var(--gradient-from, var(--ion-color-primary)),var(--gradient-to, var(--ion-color-tertiary)));box-shadow:0 8px 32px #00000026}.welcome-card--gradient .welcome-card__greeting,.welcome-card--gradient .welcome-card__subtitle{color:#ffffffd9}.welcome-card--gradient .welcome-card__name{color:#fff}.welcome-card--gradient .welcome-card__action{background:#ffffff26;color:#fff;border-color:#fff3}.welcome-card--gradient .welcome-card__action:hover{background:#ffffff40}.welcome-card--glass{background:#ffffff26;backdrop-filter:blur(20px);-webkit-backdrop-filter:blur(20px);border:1px solid rgba(255,255,255,.2);box-shadow:0 8px 32px #0000001a}.welcome-card--glass .welcome-card__greeting,.welcome-card--glass .welcome-card__subtitle{color:var(--ion-color-medium)}.welcome-card--glass .welcome-card__name{color:var(--ion-text-color)}.welcome-card--minimal{background:transparent;padding:1rem 0;border-radius:0;box-shadow:none}.welcome-card--minimal .welcome-card__greeting,.welcome-card--minimal .welcome-card__subtitle{color:var(--ion-color-medium)}.welcome-card--minimal .welcome-card__name{color:var(--ion-text-color)}.welcome-card--dark.welcome-card--default{background:var(--ion-card-background, #1e1e1e);box-shadow:0 4px 20px #0000004d}.welcome-card--dark.welcome-card--glass{background:#1e1e1eb3;border-color:#ffffff1a}.welcome-card:hover{transform:translateY(-2px)}.welcome-card--loading{pointer-events:none}.welcome-card__particles{position:absolute;inset:0;overflow:hidden;pointer-events:none}.welcome-card__particles .particle{position:absolute;border-radius:50%;background:#fff3;animation:float 8s ease-in-out infinite}.welcome-card__particles .particle--1{width:80px;height:80px;top:-20px;right:10%;animation-delay:0s}.welcome-card__particles .particle--2{width:60px;height:60px;bottom:-15px;right:25%;animation-delay:-2s}.welcome-card__particles .particle--3{width:40px;height:40px;top:40%;right:-10px;animation-delay:-4s}.welcome-card__particles .particle--4{width:30px;height:30px;top:20%;left:10%;animation-delay:-1s;opacity:.5}.welcome-card__particles .particle--5{width:50px;height:50px;bottom:10%;left:5%;animation-delay:-3s;opacity:.3}.welcome-card__particles .particle--6{width:25px;height:25px;top:60%;left:30%;animation-delay:-5s;opacity:.4}@keyframes float{0%,to{transform:translateY(0) scale(1);opacity:.2}50%{transform:translateY(-20px) scale(1.1);opacity:.35}}.welcome-card__content{position:relative;z-index:1;display:flex;align-items:center;gap:1.25rem}.welcome-card__avatar{position:relative;flex-shrink:0;display:flex;align-items:center;justify-content:center;border-radius:50%;background:var(--avatar-bg);color:var(--avatar-color);font-weight:700;transition:transform .3s ease}.welcome-card__avatar--small{width:48px;height:48px;font-size:1rem}.welcome-card__avatar--small ion-icon{font-size:1.25rem}.welcome-card__avatar--medium{width:64px;height:64px;font-size:1.25rem}.welcome-card__avatar--medium ion-icon{font-size:1.5rem}.welcome-card__avatar--large{width:80px;height:80px;font-size:1.75rem}.welcome-card__avatar--large ion-icon{font-size:2rem}.welcome-card__avatar--ring:before{content:\"\";position:absolute;inset:-4px;border-radius:50%;border:3px solid var(--ring-color, rgba(255, 255, 255, .5));animation:pulse-ring 2s ease-in-out infinite}.welcome-card__avatar .avatar-skeleton{width:100%;height:100%;border-radius:50%}.welcome-card__avatar .avatar-image{width:100%;height:100%;border-radius:50%;object-fit:cover}.welcome-card__avatar .avatar-initials{-webkit-user-select:none;user-select:none;text-transform:uppercase;letter-spacing:.5px}.welcome-card__avatar:hover{transform:scale(1.05)}@keyframes pulse-ring{0%,to{transform:scale(1);opacity:.5}50%{transform:scale(1.05);opacity:.8}}.welcome-card__text{flex:1;min-width:0}.welcome-card__greeting{display:block;font-size:.875rem;font-weight:500;margin-bottom:.25rem;text-transform:uppercase;letter-spacing:.5px}.welcome-card__name{margin:0 0 .25rem;font-size:1.5rem;font-weight:700;line-height:1.2;white-space:nowrap;overflow:hidden;text-overflow:ellipsis}.welcome-card__subtitle{display:block;font-size:.875rem;white-space:nowrap;overflow:hidden;text-overflow:ellipsis}.welcome-card__actions{position:relative;z-index:1;display:flex;flex-wrap:wrap;gap:.75rem;margin-top:1.25rem;padding-top:1rem;border-top:1px solid rgba(255,255,255,.15)}.welcome-card__action{display:inline-flex;align-items:center;gap:.5rem;padding:.5rem 1rem;border-radius:12px;border:1px solid transparent;background:var(--ion-color-light);color:var(--ion-text-color);font-size:.875rem;font-weight:500;cursor:pointer;transition:all .2s ease;text-decoration:none}.welcome-card__action ion-icon{font-size:1rem}.welcome-card__action .action-chevron{font-size:.75rem;opacity:.6;transition:transform .2s ease}.welcome-card__action:hover .action-chevron{transform:translate(3px)}@media (max-width: 480px){.welcome-card{padding:1.25rem}.welcome-card__content{gap:1rem}.welcome-card__avatar--large{width:64px;height:64px;font-size:1.25rem}.welcome-card__name{font-size:1.25rem}.welcome-card__actions{flex-direction:column}.welcome-card__action{width:100%;justify-content:space-between}}:host{animation:slideInUp .5s ease-out}@keyframes slideInUp{0%{opacity:0;transform:translateY(20px)}to{opacity:1;transform:translateY(0)}}\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"] }, { kind: "component", type: IonSkeletonText, selector: "ion-skeleton-text", inputs: ["animated"] }] }); }
294
+ }
295
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: WelcomeCardComponent, decorators: [{
296
+ type: Component,
297
+ args: [{ selector: 'val-welcome-card', standalone: true, imports: [CommonModule, RouterLink, IonIcon, IonRippleEffect, IonSkeletonText], template: `
298
+ <article
299
+ class="welcome-card"
300
+ [class.welcome-card--default]="config().variant === 'default'"
301
+ [class.welcome-card--gradient]="config().variant === 'gradient'"
302
+ [class.welcome-card--glass]="config().variant === 'glass'"
303
+ [class.welcome-card--minimal]="config().variant === 'minimal'"
304
+ [class.welcome-card--dark]="isDark()"
305
+ [class.welcome-card--loading]="config().loading"
306
+ [style.--gradient-from]="getGradientFrom()"
307
+ [style.--gradient-to]="getGradientTo()"
308
+ [style.--gradient-angle]="config().gradientAngle + 'deg'"
309
+ [style.--bg-color]="getBackgroundColor()"
310
+ (click)="onCardClick($event)"
311
+ >
312
+ <!-- Animated particles background -->
313
+ @if (config().showParticles && config().variant === 'gradient') {
314
+ <div class="welcome-card__particles">
315
+ @for (i of [1,2,3,4,5,6]; track i) {
316
+ <span class="particle" [class]="'particle--' + i"></span>
317
+ }
318
+ </div>
319
+ }
320
+
321
+ <!-- Content -->
322
+ <div class="welcome-card__content">
323
+ <!-- Avatar -->
324
+ <div
325
+ class="welcome-card__avatar"
326
+ [class.welcome-card__avatar--small]="avatarConfig().size === 'small'"
327
+ [class.welcome-card__avatar--medium]="avatarConfig().size === 'medium'"
328
+ [class.welcome-card__avatar--large]="avatarConfig().size === 'large'"
329
+ [class.welcome-card__avatar--ring]="avatarConfig().showRing"
330
+ [style.--avatar-bg]="getAvatarBackground()"
331
+ [style.--avatar-color]="getAvatarTextColor()"
332
+ [style.--ring-color]="getRingColor()"
333
+ (click)="onAvatarClick($event)"
334
+ >
335
+ @if (config().loading) {
336
+ <ion-skeleton-text [animated]="true" class="avatar-skeleton"></ion-skeleton-text>
337
+ } @else if (avatarConfig().mode === 'image' && (avatarConfig().imageUrl || config().user?.avatarUrl)) {
338
+ <img
339
+ [src]="avatarConfig().imageUrl || config().user?.avatarUrl"
340
+ [alt]="config().user?.name || 'Avatar'"
341
+ class="avatar-image"
342
+ />
343
+ } @else if (avatarConfig().mode === 'icon') {
344
+ <ion-icon [name]="avatarConfig().icon || 'person-outline'"></ion-icon>
345
+ } @else {
346
+ <span class="avatar-initials">{{ getInitials() }}</span>
347
+ }
348
+ </div>
349
+
350
+ <!-- Text content -->
351
+ <div class="welcome-card__text">
352
+ @if (config().loading) {
353
+ <ion-skeleton-text [animated]="true" style="width: 40%; height: 14px; margin-bottom: 8px;"></ion-skeleton-text>
354
+ <ion-skeleton-text [animated]="true" style="width: 60%; height: 24px; margin-bottom: 4px;"></ion-skeleton-text>
355
+ <ion-skeleton-text [animated]="true" style="width: 50%; height: 14px;"></ion-skeleton-text>
356
+ } @else {
357
+ @if (getGreeting()) {
358
+ <span class="welcome-card__greeting">{{ getGreeting() }}</span>
359
+ }
360
+ <h2 class="welcome-card__name">{{ config().user?.name || 'User' }}</h2>
361
+ @if (getSubtitle() || config().user?.email) {
362
+ <span class="welcome-card__subtitle">{{ getSubtitle() || config().user?.email }}</span>
363
+ }
364
+ }
365
+ </div>
366
+ </div>
367
+
368
+ <!-- Actions -->
369
+ @if (config().actions?.length && !config().loading) {
370
+ <div class="welcome-card__actions">
371
+ @for (action of config().actions; track action.token || action.label) {
372
+ <button
373
+ type="button"
374
+ class="welcome-card__action"
375
+ [routerLink]="action.routerLink"
376
+ (click)="onActionClick($event, action.token)"
377
+ >
378
+ @if (action.icon) {
379
+ <ion-icon [name]="action.icon"></ion-icon>
380
+ }
381
+ <span>{{ getActionLabel(action) }}</span>
382
+ <ion-icon name="chevron-forward-outline" class="action-chevron"></ion-icon>
383
+ </button>
384
+ }
385
+ </div>
386
+ }
387
+
388
+ <ion-ripple-effect></ion-ripple-effect>
389
+ </article>
390
+ `, styles: [":host{display:block;width:100%}.welcome-card{position:relative;border-radius:20px;padding:1.5rem;overflow:hidden;cursor:pointer;transition:transform .3s ease,box-shadow .3s ease}.welcome-card--default{background:var(--bg-color, var(--ion-card-background, #fff));box-shadow:0 4px 20px #00000014}.welcome-card--default .welcome-card__greeting,.welcome-card--default .welcome-card__subtitle{color:var(--ion-color-medium)}.welcome-card--default .welcome-card__name{color:var(--ion-text-color)}.welcome-card--gradient{background:linear-gradient(var(--gradient-angle, 135deg),var(--gradient-from, var(--ion-color-primary)),var(--gradient-to, var(--ion-color-tertiary)));box-shadow:0 8px 32px #00000026}.welcome-card--gradient .welcome-card__greeting,.welcome-card--gradient .welcome-card__subtitle{color:#ffffffd9}.welcome-card--gradient .welcome-card__name{color:#fff}.welcome-card--gradient .welcome-card__action{background:#ffffff26;color:#fff;border-color:#fff3}.welcome-card--gradient .welcome-card__action:hover{background:#ffffff40}.welcome-card--glass{background:#ffffff26;backdrop-filter:blur(20px);-webkit-backdrop-filter:blur(20px);border:1px solid rgba(255,255,255,.2);box-shadow:0 8px 32px #0000001a}.welcome-card--glass .welcome-card__greeting,.welcome-card--glass .welcome-card__subtitle{color:var(--ion-color-medium)}.welcome-card--glass .welcome-card__name{color:var(--ion-text-color)}.welcome-card--minimal{background:transparent;padding:1rem 0;border-radius:0;box-shadow:none}.welcome-card--minimal .welcome-card__greeting,.welcome-card--minimal .welcome-card__subtitle{color:var(--ion-color-medium)}.welcome-card--minimal .welcome-card__name{color:var(--ion-text-color)}.welcome-card--dark.welcome-card--default{background:var(--ion-card-background, #1e1e1e);box-shadow:0 4px 20px #0000004d}.welcome-card--dark.welcome-card--glass{background:#1e1e1eb3;border-color:#ffffff1a}.welcome-card:hover{transform:translateY(-2px)}.welcome-card--loading{pointer-events:none}.welcome-card__particles{position:absolute;inset:0;overflow:hidden;pointer-events:none}.welcome-card__particles .particle{position:absolute;border-radius:50%;background:#fff3;animation:float 8s ease-in-out infinite}.welcome-card__particles .particle--1{width:80px;height:80px;top:-20px;right:10%;animation-delay:0s}.welcome-card__particles .particle--2{width:60px;height:60px;bottom:-15px;right:25%;animation-delay:-2s}.welcome-card__particles .particle--3{width:40px;height:40px;top:40%;right:-10px;animation-delay:-4s}.welcome-card__particles .particle--4{width:30px;height:30px;top:20%;left:10%;animation-delay:-1s;opacity:.5}.welcome-card__particles .particle--5{width:50px;height:50px;bottom:10%;left:5%;animation-delay:-3s;opacity:.3}.welcome-card__particles .particle--6{width:25px;height:25px;top:60%;left:30%;animation-delay:-5s;opacity:.4}@keyframes float{0%,to{transform:translateY(0) scale(1);opacity:.2}50%{transform:translateY(-20px) scale(1.1);opacity:.35}}.welcome-card__content{position:relative;z-index:1;display:flex;align-items:center;gap:1.25rem}.welcome-card__avatar{position:relative;flex-shrink:0;display:flex;align-items:center;justify-content:center;border-radius:50%;background:var(--avatar-bg);color:var(--avatar-color);font-weight:700;transition:transform .3s ease}.welcome-card__avatar--small{width:48px;height:48px;font-size:1rem}.welcome-card__avatar--small ion-icon{font-size:1.25rem}.welcome-card__avatar--medium{width:64px;height:64px;font-size:1.25rem}.welcome-card__avatar--medium ion-icon{font-size:1.5rem}.welcome-card__avatar--large{width:80px;height:80px;font-size:1.75rem}.welcome-card__avatar--large ion-icon{font-size:2rem}.welcome-card__avatar--ring:before{content:\"\";position:absolute;inset:-4px;border-radius:50%;border:3px solid var(--ring-color, rgba(255, 255, 255, .5));animation:pulse-ring 2s ease-in-out infinite}.welcome-card__avatar .avatar-skeleton{width:100%;height:100%;border-radius:50%}.welcome-card__avatar .avatar-image{width:100%;height:100%;border-radius:50%;object-fit:cover}.welcome-card__avatar .avatar-initials{-webkit-user-select:none;user-select:none;text-transform:uppercase;letter-spacing:.5px}.welcome-card__avatar:hover{transform:scale(1.05)}@keyframes pulse-ring{0%,to{transform:scale(1);opacity:.5}50%{transform:scale(1.05);opacity:.8}}.welcome-card__text{flex:1;min-width:0}.welcome-card__greeting{display:block;font-size:.875rem;font-weight:500;margin-bottom:.25rem;text-transform:uppercase;letter-spacing:.5px}.welcome-card__name{margin:0 0 .25rem;font-size:1.5rem;font-weight:700;line-height:1.2;white-space:nowrap;overflow:hidden;text-overflow:ellipsis}.welcome-card__subtitle{display:block;font-size:.875rem;white-space:nowrap;overflow:hidden;text-overflow:ellipsis}.welcome-card__actions{position:relative;z-index:1;display:flex;flex-wrap:wrap;gap:.75rem;margin-top:1.25rem;padding-top:1rem;border-top:1px solid rgba(255,255,255,.15)}.welcome-card__action{display:inline-flex;align-items:center;gap:.5rem;padding:.5rem 1rem;border-radius:12px;border:1px solid transparent;background:var(--ion-color-light);color:var(--ion-text-color);font-size:.875rem;font-weight:500;cursor:pointer;transition:all .2s ease;text-decoration:none}.welcome-card__action ion-icon{font-size:1rem}.welcome-card__action .action-chevron{font-size:.75rem;opacity:.6;transition:transform .2s ease}.welcome-card__action:hover .action-chevron{transform:translate(3px)}@media (max-width: 480px){.welcome-card{padding:1.25rem}.welcome-card__content{gap:1rem}.welcome-card__avatar--large{width:64px;height:64px;font-size:1.25rem}.welcome-card__name{font-size:1.25rem}.welcome-card__actions{flex-direction:column}.welcome-card__action{width:100%;justify-content:space-between}}:host{animation:slideInUp .5s ease-out}@keyframes slideInUp{0%{opacity:0;transform:translateY(20px)}to{opacity:1;transform:translateY(0)}}\n"] }]
391
+ }], propDecorators: { cardClick: [{
392
+ type: Output
393
+ }] } });
394
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"welcome-card.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/molecules/welcome-card/welcome-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, IonSkeletonText } from '@ionic/angular/standalone';\nimport { addIcons } from 'ionicons';\nimport { personOutline, chevronForwardOutline } from 'ionicons/icons';\nimport { I18nService } from '../../../services/i18n';\nimport { ThemeService } from '../../../services/theme.service';\nimport {\n  WelcomeCardMetadata,\n  WelcomeCardClickEvent,\n  WELCOME_CARD_DEFAULTS,\n} from './types';\n\naddIcons({ personOutline, chevronForwardOutline });\n\nconst IONIC_COLORS = [\n  'primary', 'secondary', 'tertiary', 'success',\n  'warning', 'danger', 'light', 'medium', 'dark'\n];\n\n// Vibrant color palette for avatar backgrounds\nconst AVATAR_COLORS = [\n  { bg: '#6366f1', text: '#ffffff' }, // Indigo\n  { bg: '#8b5cf6', text: '#ffffff' }, // Violet\n  { bg: '#ec4899', text: '#ffffff' }, // Pink\n  { bg: '#f43f5e', text: '#ffffff' }, // Rose\n  { bg: '#f97316', text: '#ffffff' }, // Orange\n  { bg: '#eab308', text: '#1f2937' }, // Yellow\n  { bg: '#22c55e', text: '#ffffff' }, // Green\n  { bg: '#14b8a6', text: '#ffffff' }, // Teal\n  { bg: '#06b6d4', text: '#ffffff' }, // Cyan\n  { bg: '#3b82f6', text: '#ffffff' }, // Blue\n];\n\n/**\n * val-welcome-card\n *\n * A modern welcome/greeting card with avatar, animated gradient background,\n * and optional action buttons. Perfect for dashboard headers.\n *\n * @example Basic usage\n * ```html\n * <val-welcome-card\n *   [props]=\"{\n *     greeting: 'Welcome back',\n *     user: { name: 'John Doe', email: 'john@example.com' }\n *   }\"\n * />\n * ```\n *\n * @example With gradient variant\n * ```html\n * <val-welcome-card\n *   [props]=\"{\n *     variant: 'gradient',\n *     greeting: 'Good morning',\n *     user: { name: 'Jane', email: 'jane@example.com' },\n *     gradientFrom: 'primary',\n *     gradientTo: 'tertiary'\n *   }\"\n * />\n * ```\n *\n * @example With i18n\n * ```html\n * <val-welcome-card\n *   [props]=\"{\n *     greetingKey: 'welcome',\n *     i18nNamespace: 'Dashboard',\n *     user: user()\n *   }\"\n * />\n * ```\n */\n@Component({\n  selector: 'val-welcome-card',\n  standalone: true,\n  imports: [CommonModule, RouterLink, IonIcon, IonRippleEffect, IonSkeletonText],\n  template: `\n    <article\n      class=\"welcome-card\"\n      [class.welcome-card--default]=\"config().variant === 'default'\"\n      [class.welcome-card--gradient]=\"config().variant === 'gradient'\"\n      [class.welcome-card--glass]=\"config().variant === 'glass'\"\n      [class.welcome-card--minimal]=\"config().variant === 'minimal'\"\n      [class.welcome-card--dark]=\"isDark()\"\n      [class.welcome-card--loading]=\"config().loading\"\n      [style.--gradient-from]=\"getGradientFrom()\"\n      [style.--gradient-to]=\"getGradientTo()\"\n      [style.--gradient-angle]=\"config().gradientAngle + 'deg'\"\n      [style.--bg-color]=\"getBackgroundColor()\"\n      (click)=\"onCardClick($event)\"\n    >\n      <!-- Animated particles background -->\n      @if (config().showParticles && config().variant === 'gradient') {\n        <div class=\"welcome-card__particles\">\n          @for (i of [1,2,3,4,5,6]; track i) {\n            <span class=\"particle\" [class]=\"'particle--' + i\"></span>\n          }\n        </div>\n      }\n\n      <!-- Content -->\n      <div class=\"welcome-card__content\">\n        <!-- Avatar -->\n        <div\n          class=\"welcome-card__avatar\"\n          [class.welcome-card__avatar--small]=\"avatarConfig().size === 'small'\"\n          [class.welcome-card__avatar--medium]=\"avatarConfig().size === 'medium'\"\n          [class.welcome-card__avatar--large]=\"avatarConfig().size === 'large'\"\n          [class.welcome-card__avatar--ring]=\"avatarConfig().showRing\"\n          [style.--avatar-bg]=\"getAvatarBackground()\"\n          [style.--avatar-color]=\"getAvatarTextColor()\"\n          [style.--ring-color]=\"getRingColor()\"\n          (click)=\"onAvatarClick($event)\"\n        >\n          @if (config().loading) {\n            <ion-skeleton-text [animated]=\"true\" class=\"avatar-skeleton\"></ion-skeleton-text>\n          } @else if (avatarConfig().mode === 'image' && (avatarConfig().imageUrl || config().user?.avatarUrl)) {\n            <img\n              [src]=\"avatarConfig().imageUrl || config().user?.avatarUrl\"\n              [alt]=\"config().user?.name || 'Avatar'\"\n              class=\"avatar-image\"\n            />\n          } @else if (avatarConfig().mode === 'icon') {\n            <ion-icon [name]=\"avatarConfig().icon || 'person-outline'\"></ion-icon>\n          } @else {\n            <span class=\"avatar-initials\">{{ getInitials() }}</span>\n          }\n        </div>\n\n        <!-- Text content -->\n        <div class=\"welcome-card__text\">\n          @if (config().loading) {\n            <ion-skeleton-text [animated]=\"true\" style=\"width: 40%; height: 14px; margin-bottom: 8px;\"></ion-skeleton-text>\n            <ion-skeleton-text [animated]=\"true\" style=\"width: 60%; height: 24px; margin-bottom: 4px;\"></ion-skeleton-text>\n            <ion-skeleton-text [animated]=\"true\" style=\"width: 50%; height: 14px;\"></ion-skeleton-text>\n          } @else {\n            @if (getGreeting()) {\n              <span class=\"welcome-card__greeting\">{{ getGreeting() }}</span>\n            }\n            <h2 class=\"welcome-card__name\">{{ config().user?.name || 'User' }}</h2>\n            @if (getSubtitle() || config().user?.email) {\n              <span class=\"welcome-card__subtitle\">{{ getSubtitle() || config().user?.email }}</span>\n            }\n          }\n        </div>\n      </div>\n\n      <!-- Actions -->\n      @if (config().actions?.length && !config().loading) {\n        <div class=\"welcome-card__actions\">\n          @for (action of config().actions; track action.token || action.label) {\n            <button\n              type=\"button\"\n              class=\"welcome-card__action\"\n              [routerLink]=\"action.routerLink\"\n              (click)=\"onActionClick($event, action.token)\"\n            >\n              @if (action.icon) {\n                <ion-icon [name]=\"action.icon\"></ion-icon>\n              }\n              <span>{{ getActionLabel(action) }}</span>\n              <ion-icon name=\"chevron-forward-outline\" class=\"action-chevron\"></ion-icon>\n            </button>\n          }\n        </div>\n      }\n\n      <ion-ripple-effect></ion-ripple-effect>\n    </article>\n  `,\n  styleUrls: ['./welcome-card.component.scss'],\n})\nexport class WelcomeCardComponent {\n  private i18n = inject(I18nService);\n  private themeService = inject(ThemeService);\n\n  /** Component configuration */\n  readonly props = input<Partial<WelcomeCardMetadata>>({});\n\n  /** Event emitted when card elements are clicked */\n  @Output() cardClick = new EventEmitter<WelcomeCardClickEvent>();\n\n  /** Merged configuration with defaults */\n  config = computed(() => ({\n    ...WELCOME_CARD_DEFAULTS,\n    ...this.props(),\n  }));\n\n  /** Avatar configuration with defaults */\n  avatarConfig = computed(() => ({\n    ...WELCOME_CARD_DEFAULTS.avatar,\n    ...this.props().avatar,\n  }));\n\n  /** Current theme mode */\n  isDark = computed(() => this.themeService.IsDark);\n\n  /** Color index based on user name hash */\n  private colorIndex = computed(() => {\n    const name = this.config().user?.name || '';\n    let hash = 0;\n    for (let i = 0; i < name.length; i++) {\n      hash = name.charCodeAt(i) + ((hash << 5) - hash);\n    }\n    return Math.abs(hash) % AVATAR_COLORS.length;\n  });\n\n  /** Get initials from user name */\n  getInitials(): string {\n    const props = this.props();\n    // Check for explicit initials first (from props, not merged config)\n    if (props.avatar?.initials || props.user?.initials) {\n      return props.avatar?.initials || props.user?.initials || '';\n    }\n\n    const name = props.user?.name || '';\n    if (!name) return '?';\n\n    const parts = name.trim().split(/\\s+/);\n    if (parts.length === 1) {\n      return parts[0].substring(0, 2).toUpperCase();\n    }\n    return (parts[0][0] + parts[parts.length - 1][0]).toUpperCase();\n  }\n\n  /** Get greeting text with i18n support */\n  getGreeting(): string {\n    const cfg = this.config();\n    if (cfg.i18nNamespace && cfg.greetingKey) {\n      this.i18n.lang(); // Track for reactivity\n      return this.i18n.t(cfg.greetingKey, cfg.i18nNamespace);\n    }\n    return cfg.greeting || '';\n  }\n\n  /** Get subtitle text with i18n support */\n  getSubtitle(): string {\n    const cfg = this.config();\n    if (cfg.i18nNamespace && cfg.subtitleKey) {\n      this.i18n.lang();\n      return this.i18n.t(cfg.subtitleKey, cfg.i18nNamespace);\n    }\n    return cfg.subtitle || '';\n  }\n\n  /** Get action label with i18n support */\n  getActionLabel(action: { label?: string; labelKey?: string }): string {\n    const cfg = this.config();\n    if (cfg.i18nNamespace && action.labelKey) {\n      this.i18n.lang();\n      return this.i18n.t(action.labelKey, cfg.i18nNamespace);\n    }\n    return action.label || '';\n  }\n\n  /** Resolve color 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  /** Get gradient start color */\n  getGradientFrom(): string {\n    const color = this.config().gradientFrom;\n    return this.resolveColor(color as string) || 'var(--ion-color-primary)';\n  }\n\n  /** Get gradient end color */\n  getGradientTo(): string {\n    const color = this.config().gradientTo;\n    return this.resolveColor(color as string) || 'var(--ion-color-tertiary)';\n  }\n\n  /** Get background color (for non-gradient variants) */\n  getBackgroundColor(): string | null {\n    return this.resolveColor(this.config().backgroundColor as string);\n  }\n\n  /** Get avatar background based on user name */\n  getAvatarBackground(): string {\n    const cfg = this.avatarConfig();\n    if (cfg.mode === 'image') return 'transparent';\n\n    return AVATAR_COLORS[this.colorIndex()].bg;\n  }\n\n  /** Get avatar text color */\n  getAvatarTextColor(): string {\n    return AVATAR_COLORS[this.colorIndex()].text;\n  }\n\n  /** Get ring color */\n  getRingColor(): string {\n    const ringColor = this.avatarConfig().ringColor;\n    return this.resolveColor(ringColor as string) || 'rgba(255, 255, 255, 0.5)';\n  }\n\n  /** Handle card click */\n  onCardClick(event: MouseEvent): void {\n    // Don't emit if clicking avatar or actions\n    if ((event.target as HTMLElement).closest('.welcome-card__avatar, .welcome-card__action')) {\n      return;\n    }\n    this.cardClick.emit({ target: 'card' });\n  }\n\n  /** Handle avatar click */\n  onAvatarClick(event: MouseEvent): void {\n    event.stopPropagation();\n    this.cardClick.emit({ target: 'avatar' });\n  }\n\n  /** Handle action click */\n  onActionClick(event: MouseEvent, token?: string): void {\n    event.stopPropagation();\n    this.cardClick.emit({ target: 'action', actionToken: token });\n  }\n}\n"]}