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.
- package/esm2022/lib/components/molecules/welcome-card/types.mjs +15 -0
- package/esm2022/lib/components/molecules/welcome-card/welcome-card.component.mjs +394 -0
- package/esm2022/public-api.mjs +3 -1
- package/fesm2022/valtech-components.mjs +402 -3
- package/fesm2022/valtech-components.mjs.map +1 -1
- package/lib/components/molecules/welcome-card/types.d.ts +114 -0
- package/lib/components/molecules/welcome-card/welcome-card.component.d.ts +117 -0
- package/package.json +1 -1
- package/public-api.d.ts +2 -0
|
@@ -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"]}
|