valtech-components 2.0.510 → 2.0.512
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/refresher/refresher.component.mjs +254 -0
- package/esm2022/lib/components/molecules/refresher/types.mjs +15 -0
- package/esm2022/lib/components/organisms/infinite-list/infinite-list.component.mjs +618 -0
- package/esm2022/lib/components/organisms/infinite-list/types.mjs +15 -0
- package/esm2022/lib/components/templates/page-template/page-template.component.mjs +5 -5
- package/esm2022/lib/services/pagination/index.mjs +5 -0
- package/esm2022/lib/services/pagination/pagination.service.mjs +218 -0
- package/esm2022/lib/services/pagination/types.mjs +14 -0
- package/esm2022/lib/services/skeleton/config.mjs +79 -0
- package/esm2022/lib/services/skeleton/directives/loading.directive.mjs +215 -0
- package/esm2022/lib/services/skeleton/index.mjs +16 -0
- package/esm2022/lib/services/skeleton/skeleton.service.mjs +198 -0
- package/esm2022/lib/services/skeleton/templates/detail-skeleton.component.mjs +223 -0
- package/esm2022/lib/services/skeleton/templates/form-skeleton.component.mjs +127 -0
- package/esm2022/lib/services/skeleton/templates/grid-skeleton.component.mjs +154 -0
- package/esm2022/lib/services/skeleton/templates/list-skeleton.component.mjs +110 -0
- package/esm2022/lib/services/skeleton/templates/profile-skeleton.component.mjs +207 -0
- package/esm2022/lib/services/skeleton/templates/table-skeleton.component.mjs +116 -0
- package/esm2022/lib/services/skeleton/types.mjs +11 -0
- package/esm2022/public-api.mjs +12 -1
- package/fesm2022/valtech-components.mjs +3887 -1370
- package/fesm2022/valtech-components.mjs.map +1 -1
- package/lib/components/molecules/refresher/refresher.component.d.ts +79 -0
- package/lib/components/molecules/refresher/types.d.ts +86 -0
- package/lib/components/organisms/infinite-list/infinite-list.component.d.ts +111 -0
- package/lib/components/organisms/infinite-list/types.d.ts +197 -0
- package/lib/services/pagination/index.d.ts +2 -0
- package/lib/services/pagination/pagination.service.d.ts +43 -0
- package/lib/services/pagination/types.d.ts +113 -0
- package/lib/services/skeleton/config.d.ts +30 -0
- package/lib/services/skeleton/directives/loading.directive.d.ts +71 -0
- package/lib/services/skeleton/index.d.ts +10 -0
- package/lib/services/skeleton/skeleton.service.d.ts +127 -0
- package/lib/services/skeleton/templates/detail-skeleton.component.d.ts +18 -0
- package/lib/services/skeleton/templates/form-skeleton.component.d.ts +22 -0
- package/lib/services/skeleton/templates/grid-skeleton.component.d.ts +18 -0
- package/lib/services/skeleton/templates/list-skeleton.component.d.ts +17 -0
- package/lib/services/skeleton/templates/profile-skeleton.component.d.ts +20 -0
- package/lib/services/skeleton/templates/table-skeleton.component.d.ts +17 -0
- package/lib/services/skeleton/types.d.ts +111 -0
- package/package.json +1 -1
- package/public-api.d.ts +6 -0
|
@@ -0,0 +1,215 @@
|
|
|
1
|
+
import { Directive, Input, TemplateRef, ViewContainerRef, inject, effect, signal, isSignal, DestroyRef, } from '@angular/core';
|
|
2
|
+
import { toSignal } from '@angular/core/rxjs-interop';
|
|
3
|
+
import { isObservable } from 'rxjs';
|
|
4
|
+
import { SkeletonService } from '../skeleton.service';
|
|
5
|
+
import * as i0 from "@angular/core";
|
|
6
|
+
/**
|
|
7
|
+
* Directiva estructural simplificada para estados de carga.
|
|
8
|
+
*
|
|
9
|
+
* Soporta multiples fuentes de estado de carga:
|
|
10
|
+
* - Angular Signal<boolean>
|
|
11
|
+
* - RxJS Observable<boolean>
|
|
12
|
+
* - Promise<any> (cargando hasta que se resuelve)
|
|
13
|
+
* - boolean literal
|
|
14
|
+
*
|
|
15
|
+
* @example
|
|
16
|
+
* <!-- Uso simple con signal -->
|
|
17
|
+
* <ng-container *valLoading="isLoading(); skeleton: 'list'">
|
|
18
|
+
* <app-list [items]="items()"></app-list>
|
|
19
|
+
* </ng-container>
|
|
20
|
+
*
|
|
21
|
+
* @example
|
|
22
|
+
* <!-- Con configuracion -->
|
|
23
|
+
* <ng-container *valLoading="loading$; skeleton: 'grid-cards'; count: 8">
|
|
24
|
+
* <app-grid [data]="data"></app-grid>
|
|
25
|
+
* </ng-container>
|
|
26
|
+
*
|
|
27
|
+
* @example
|
|
28
|
+
* <!-- Con template personalizado -->
|
|
29
|
+
* <ng-container *valLoading="isLoading(); skeletonTpl: customSkeleton">
|
|
30
|
+
* <app-content></app-content>
|
|
31
|
+
* </ng-container>
|
|
32
|
+
* <ng-template #customSkeleton>
|
|
33
|
+
* <val-skeleton [props]="{ type: 'card' }"></val-skeleton>
|
|
34
|
+
* </ng-template>
|
|
35
|
+
*/
|
|
36
|
+
export class LoadingDirective {
|
|
37
|
+
constructor() {
|
|
38
|
+
this.templateRef = inject((TemplateRef));
|
|
39
|
+
this.viewContainer = inject(ViewContainerRef);
|
|
40
|
+
this.skeletonService = inject(SkeletonService);
|
|
41
|
+
this.destroyRef = inject(DestroyRef);
|
|
42
|
+
this._loading = signal(false);
|
|
43
|
+
this.hasContentView = false;
|
|
44
|
+
this.skeletonComponentRef = null;
|
|
45
|
+
/** Template de skeleton a usar */
|
|
46
|
+
this.skeleton = 'list';
|
|
47
|
+
/** Template personalizado para skeleton */
|
|
48
|
+
this.skeletonTpl = null;
|
|
49
|
+
/** Cantidad de items skeleton */
|
|
50
|
+
this.count = 3;
|
|
51
|
+
/** Animacion habilitada */
|
|
52
|
+
this.animated = true;
|
|
53
|
+
/** Mostrar spinner en lugar de skeleton */
|
|
54
|
+
this.spinner = false;
|
|
55
|
+
effect(() => {
|
|
56
|
+
const isLoading = this._loading();
|
|
57
|
+
this.updateView(isLoading);
|
|
58
|
+
});
|
|
59
|
+
}
|
|
60
|
+
/**
|
|
61
|
+
* Input principal - puede ser boolean, Signal, Observable o Promise.
|
|
62
|
+
*/
|
|
63
|
+
set loading(source) {
|
|
64
|
+
this.resolveLoadingSource(source);
|
|
65
|
+
}
|
|
66
|
+
ngOnDestroy() {
|
|
67
|
+
this.viewContainer.clear();
|
|
68
|
+
this.skeletonComponentRef = null;
|
|
69
|
+
}
|
|
70
|
+
resolveLoadingSource(source) {
|
|
71
|
+
if (typeof source === 'boolean') {
|
|
72
|
+
this._loading.set(source);
|
|
73
|
+
}
|
|
74
|
+
else if (isSignal(source)) {
|
|
75
|
+
// Sincronizar signal con effect
|
|
76
|
+
effect(() => {
|
|
77
|
+
this._loading.set(source());
|
|
78
|
+
}, { injector: this.viewContainer.injector });
|
|
79
|
+
}
|
|
80
|
+
else if (isObservable(source)) {
|
|
81
|
+
// Convertir observable a signal
|
|
82
|
+
const signalValue = toSignal(source, {
|
|
83
|
+
initialValue: true,
|
|
84
|
+
injector: this.viewContainer.injector,
|
|
85
|
+
});
|
|
86
|
+
effect(() => {
|
|
87
|
+
this._loading.set(signalValue());
|
|
88
|
+
}, { injector: this.viewContainer.injector });
|
|
89
|
+
}
|
|
90
|
+
else if (source instanceof Promise) {
|
|
91
|
+
this._loading.set(true);
|
|
92
|
+
source.finally(() => this._loading.set(false));
|
|
93
|
+
}
|
|
94
|
+
}
|
|
95
|
+
updateView(isLoading) {
|
|
96
|
+
if (isLoading) {
|
|
97
|
+
this.showSkeleton();
|
|
98
|
+
}
|
|
99
|
+
else {
|
|
100
|
+
this.showContent();
|
|
101
|
+
}
|
|
102
|
+
}
|
|
103
|
+
showSkeleton() {
|
|
104
|
+
// Limpiar contenido previo
|
|
105
|
+
this.viewContainer.clear();
|
|
106
|
+
this.hasContentView = false;
|
|
107
|
+
if (this.skeletonTpl) {
|
|
108
|
+
// Usar template personalizado
|
|
109
|
+
this.viewContainer.createEmbeddedView(this.skeletonTpl);
|
|
110
|
+
}
|
|
111
|
+
else if (this.spinner) {
|
|
112
|
+
// Mostrar spinner (usar val-content-loader si esta disponible)
|
|
113
|
+
this.showSpinner();
|
|
114
|
+
}
|
|
115
|
+
else {
|
|
116
|
+
// Usar template de skeleton registrado
|
|
117
|
+
this.showSkeletonTemplate();
|
|
118
|
+
}
|
|
119
|
+
}
|
|
120
|
+
showSkeletonTemplate() {
|
|
121
|
+
const template = this.skeletonService.getTemplate(this.skeleton);
|
|
122
|
+
if (template) {
|
|
123
|
+
this.skeletonComponentRef = this.viewContainer.createComponent(template.component);
|
|
124
|
+
// Combinar config por defecto con inputs
|
|
125
|
+
const config = {
|
|
126
|
+
...template.defaultConfig,
|
|
127
|
+
count: this.count,
|
|
128
|
+
animated: this.animated,
|
|
129
|
+
gap: this.gap,
|
|
130
|
+
variant: this.variant,
|
|
131
|
+
};
|
|
132
|
+
this.skeletonComponentRef.instance.config = config;
|
|
133
|
+
}
|
|
134
|
+
else {
|
|
135
|
+
// Fallback: mostrar texto de carga si no hay template
|
|
136
|
+
console.warn(`[valLoading] Template '${this.skeleton}' not found. Using fallback.`);
|
|
137
|
+
this.showFallback();
|
|
138
|
+
}
|
|
139
|
+
}
|
|
140
|
+
showSpinner() {
|
|
141
|
+
// Crear un div con spinner simple
|
|
142
|
+
const element = document.createElement('div');
|
|
143
|
+
element.className = 'val-loading-spinner';
|
|
144
|
+
element.innerHTML = `
|
|
145
|
+
<ion-spinner name="crescent"></ion-spinner>
|
|
146
|
+
`;
|
|
147
|
+
element.style.cssText = 'display: flex; justify-content: center; padding: 20px;';
|
|
148
|
+
const hostElement = this.viewContainer.element.nativeElement;
|
|
149
|
+
hostElement.parentElement?.insertBefore(element, hostElement);
|
|
150
|
+
}
|
|
151
|
+
showFallback() {
|
|
152
|
+
// Fallback simple si no hay template registrado
|
|
153
|
+
const element = document.createElement('div');
|
|
154
|
+
element.className = 'val-loading-fallback';
|
|
155
|
+
element.style.cssText =
|
|
156
|
+
'display: flex; flex-direction: column; gap: 12px; padding: 16px;';
|
|
157
|
+
for (let i = 0; i < this.count; i++) {
|
|
158
|
+
const skeleton = document.createElement('div');
|
|
159
|
+
skeleton.style.cssText = `
|
|
160
|
+
background: linear-gradient(90deg, #f0f0f0 25%, #e0e0e0 50%, #f0f0f0 75%);
|
|
161
|
+
background-size: 200% 100%;
|
|
162
|
+
animation: skeleton-loading 1.5s infinite;
|
|
163
|
+
height: 56px;
|
|
164
|
+
border-radius: 8px;
|
|
165
|
+
`;
|
|
166
|
+
element.appendChild(skeleton);
|
|
167
|
+
}
|
|
168
|
+
const hostElement = this.viewContainer.element.nativeElement;
|
|
169
|
+
hostElement.parentElement?.insertBefore(element, hostElement);
|
|
170
|
+
}
|
|
171
|
+
showContent() {
|
|
172
|
+
// Limpiar skeleton
|
|
173
|
+
this.viewContainer.clear();
|
|
174
|
+
this.skeletonComponentRef = null;
|
|
175
|
+
// Mostrar contenido real
|
|
176
|
+
if (!this.hasContentView) {
|
|
177
|
+
this.viewContainer.createEmbeddedView(this.templateRef);
|
|
178
|
+
this.hasContentView = true;
|
|
179
|
+
}
|
|
180
|
+
}
|
|
181
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: LoadingDirective, deps: [], target: i0.ɵɵFactoryTarget.Directive }); }
|
|
182
|
+
static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "18.2.14", type: LoadingDirective, isStandalone: true, selector: "[valLoading]", inputs: { skeleton: ["valLoadingSkeleton", "skeleton"], skeletonTpl: ["valLoadingSkeletonTpl", "skeletonTpl"], count: ["valLoadingCount", "count"], animated: ["valLoadingAnimated", "animated"], gap: ["valLoadingGap", "gap"], variant: ["valLoadingVariant", "variant"], spinner: ["valLoadingSpinner", "spinner"], loading: ["valLoading", "loading"] }, ngImport: i0 }); }
|
|
183
|
+
}
|
|
184
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: LoadingDirective, decorators: [{
|
|
185
|
+
type: Directive,
|
|
186
|
+
args: [{
|
|
187
|
+
selector: '[valLoading]',
|
|
188
|
+
standalone: true,
|
|
189
|
+
}]
|
|
190
|
+
}], ctorParameters: () => [], propDecorators: { skeleton: [{
|
|
191
|
+
type: Input,
|
|
192
|
+
args: ['valLoadingSkeleton']
|
|
193
|
+
}], skeletonTpl: [{
|
|
194
|
+
type: Input,
|
|
195
|
+
args: ['valLoadingSkeletonTpl']
|
|
196
|
+
}], count: [{
|
|
197
|
+
type: Input,
|
|
198
|
+
args: ['valLoadingCount']
|
|
199
|
+
}], animated: [{
|
|
200
|
+
type: Input,
|
|
201
|
+
args: ['valLoadingAnimated']
|
|
202
|
+
}], gap: [{
|
|
203
|
+
type: Input,
|
|
204
|
+
args: ['valLoadingGap']
|
|
205
|
+
}], variant: [{
|
|
206
|
+
type: Input,
|
|
207
|
+
args: ['valLoadingVariant']
|
|
208
|
+
}], spinner: [{
|
|
209
|
+
type: Input,
|
|
210
|
+
args: ['valLoadingSpinner']
|
|
211
|
+
}], loading: [{
|
|
212
|
+
type: Input,
|
|
213
|
+
args: ['valLoading']
|
|
214
|
+
}] } });
|
|
215
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"loading.directive.js","sourceRoot":"","sources":["../../../../../../../src/lib/services/skeleton/directives/loading.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Directive,\n  Input,\n  TemplateRef,\n  ViewContainerRef,\n  inject,\n  effect,\n  signal,\n  OnDestroy,\n  isSignal,\n  DestroyRef,\n  ComponentRef,\n} from '@angular/core';\nimport { toSignal } from '@angular/core/rxjs-interop';\nimport { isObservable } from 'rxjs';\nimport { SkeletonService } from '../skeleton.service';\nimport { LoadingSource, SkeletonTemplateName, SkeletonTemplateComponent } from '../types';\n\n/**\n * Directiva estructural simplificada para estados de carga.\n *\n * Soporta multiples fuentes de estado de carga:\n * - Angular Signal<boolean>\n * - RxJS Observable<boolean>\n * - Promise<any> (cargando hasta que se resuelve)\n * - boolean literal\n *\n * @example\n * <!-- Uso simple con signal -->\n * <ng-container *valLoading=\"isLoading(); skeleton: 'list'\">\n *   <app-list [items]=\"items()\"></app-list>\n * </ng-container>\n *\n * @example\n * <!-- Con configuracion -->\n * <ng-container *valLoading=\"loading$; skeleton: 'grid-cards'; count: 8\">\n *   <app-grid [data]=\"data\"></app-grid>\n * </ng-container>\n *\n * @example\n * <!-- Con template personalizado -->\n * <ng-container *valLoading=\"isLoading(); skeletonTpl: customSkeleton\">\n *   <app-content></app-content>\n * </ng-container>\n * <ng-template #customSkeleton>\n *   <val-skeleton [props]=\"{ type: 'card' }\"></val-skeleton>\n * </ng-template>\n */\n@Directive({\n  selector: '[valLoading]',\n  standalone: true,\n})\nexport class LoadingDirective implements OnDestroy {\n  private readonly templateRef = inject(TemplateRef<unknown>);\n  private readonly viewContainer = inject(ViewContainerRef);\n  private readonly skeletonService = inject(SkeletonService);\n  private readonly destroyRef = inject(DestroyRef);\n\n  private readonly _loading = signal(false);\n  private hasContentView = false;\n  private skeletonComponentRef: ComponentRef<SkeletonTemplateComponent> | null = null;\n\n  /** Template de skeleton a usar */\n  @Input('valLoadingSkeleton') skeleton: SkeletonTemplateName | string = 'list';\n\n  /** Template personalizado para skeleton */\n  @Input('valLoadingSkeletonTpl') skeletonTpl: TemplateRef<unknown> | null = null;\n\n  /** Cantidad de items skeleton */\n  @Input('valLoadingCount') count = 3;\n\n  /** Animacion habilitada */\n  @Input('valLoadingAnimated') animated = true;\n\n  /** Gap entre items */\n  @Input('valLoadingGap') gap: string | undefined;\n\n  /** Variante del template */\n  @Input('valLoadingVariant') variant: string | undefined;\n\n  /** Mostrar spinner en lugar de skeleton */\n  @Input('valLoadingSpinner') spinner = false;\n\n  constructor() {\n    effect(() => {\n      const isLoading = this._loading();\n      this.updateView(isLoading);\n    });\n  }\n\n  /**\n   * Input principal - puede ser boolean, Signal, Observable o Promise.\n   */\n  @Input('valLoading')\n  set loading(source: LoadingSource) {\n    this.resolveLoadingSource(source);\n  }\n\n  ngOnDestroy(): void {\n    this.viewContainer.clear();\n    this.skeletonComponentRef = null;\n  }\n\n  private resolveLoadingSource(source: LoadingSource): void {\n    if (typeof source === 'boolean') {\n      this._loading.set(source);\n    } else if (isSignal(source)) {\n      // Sincronizar signal con effect\n      effect(\n        () => {\n          this._loading.set((source as () => boolean)());\n        },\n        { injector: this.viewContainer.injector }\n      );\n    } else if (isObservable(source)) {\n      // Convertir observable a signal\n      const signalValue = toSignal(source, {\n        initialValue: true,\n        injector: this.viewContainer.injector,\n      });\n      effect(\n        () => {\n          this._loading.set(signalValue());\n        },\n        { injector: this.viewContainer.injector }\n      );\n    } else if (source instanceof Promise) {\n      this._loading.set(true);\n      source.finally(() => this._loading.set(false));\n    }\n  }\n\n  private updateView(isLoading: boolean): void {\n    if (isLoading) {\n      this.showSkeleton();\n    } else {\n      this.showContent();\n    }\n  }\n\n  private showSkeleton(): void {\n    // Limpiar contenido previo\n    this.viewContainer.clear();\n    this.hasContentView = false;\n\n    if (this.skeletonTpl) {\n      // Usar template personalizado\n      this.viewContainer.createEmbeddedView(this.skeletonTpl);\n    } else if (this.spinner) {\n      // Mostrar spinner (usar val-content-loader si esta disponible)\n      this.showSpinner();\n    } else {\n      // Usar template de skeleton registrado\n      this.showSkeletonTemplate();\n    }\n  }\n\n  private showSkeletonTemplate(): void {\n    const template = this.skeletonService.getTemplate(this.skeleton);\n\n    if (template) {\n      this.skeletonComponentRef = this.viewContainer.createComponent(\n        template.component as new () => SkeletonTemplateComponent\n      );\n\n      // Combinar config por defecto con inputs\n      const config = {\n        ...template.defaultConfig,\n        count: this.count,\n        animated: this.animated,\n        gap: this.gap,\n        variant: this.variant,\n      };\n\n      this.skeletonComponentRef.instance.config = config;\n    } else {\n      // Fallback: mostrar texto de carga si no hay template\n      console.warn(`[valLoading] Template '${this.skeleton}' not found. Using fallback.`);\n      this.showFallback();\n    }\n  }\n\n  private showSpinner(): void {\n    // Crear un div con spinner simple\n    const element = document.createElement('div');\n    element.className = 'val-loading-spinner';\n    element.innerHTML = `\n      <ion-spinner name=\"crescent\"></ion-spinner>\n    `;\n    element.style.cssText = 'display: flex; justify-content: center; padding: 20px;';\n\n    const hostElement = this.viewContainer.element.nativeElement as HTMLElement;\n    hostElement.parentElement?.insertBefore(element, hostElement);\n  }\n\n  private showFallback(): void {\n    // Fallback simple si no hay template registrado\n    const element = document.createElement('div');\n    element.className = 'val-loading-fallback';\n    element.style.cssText =\n      'display: flex; flex-direction: column; gap: 12px; padding: 16px;';\n\n    for (let i = 0; i < this.count; i++) {\n      const skeleton = document.createElement('div');\n      skeleton.style.cssText = `\n        background: linear-gradient(90deg, #f0f0f0 25%, #e0e0e0 50%, #f0f0f0 75%);\n        background-size: 200% 100%;\n        animation: skeleton-loading 1.5s infinite;\n        height: 56px;\n        border-radius: 8px;\n      `;\n      element.appendChild(skeleton);\n    }\n\n    const hostElement = this.viewContainer.element.nativeElement as HTMLElement;\n    hostElement.parentElement?.insertBefore(element, hostElement);\n  }\n\n  private showContent(): void {\n    // Limpiar skeleton\n    this.viewContainer.clear();\n    this.skeletonComponentRef = null;\n\n    // Mostrar contenido real\n    if (!this.hasContentView) {\n      this.viewContainer.createEmbeddedView(this.templateRef);\n      this.hasContentView = true;\n    }\n  }\n}\n"]}
|
|
@@ -0,0 +1,16 @@
|
|
|
1
|
+
// Types
|
|
2
|
+
export { DEFAULT_SKELETON_CONFIG, } from './types';
|
|
3
|
+
// Service
|
|
4
|
+
export { SkeletonService } from './skeleton.service';
|
|
5
|
+
// Directives
|
|
6
|
+
export { LoadingDirective } from './directives/loading.directive';
|
|
7
|
+
// Templates
|
|
8
|
+
export { ListSkeletonComponent } from './templates/list-skeleton.component';
|
|
9
|
+
export { GridSkeletonComponent } from './templates/grid-skeleton.component';
|
|
10
|
+
export { FormSkeletonComponent } from './templates/form-skeleton.component';
|
|
11
|
+
export { ProfileSkeletonComponent } from './templates/profile-skeleton.component';
|
|
12
|
+
export { TableSkeletonComponent } from './templates/table-skeleton.component';
|
|
13
|
+
export { DetailSkeletonComponent } from './templates/detail-skeleton.component';
|
|
14
|
+
// Config / Provider
|
|
15
|
+
export { provideValtechSkeleton } from './config';
|
|
16
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9zcmMvbGliL3NlcnZpY2VzL3NrZWxldG9uL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLFFBQVE7QUFDUixPQUFPLEVBVUwsdUJBQXVCLEdBQ3hCLE1BQU0sU0FBUyxDQUFDO0FBRWpCLFVBQVU7QUFDVixPQUFPLEVBQUUsZUFBZSxFQUFFLE1BQU0sb0JBQW9CLENBQUM7QUFFckQsYUFBYTtBQUNiLE9BQU8sRUFBRSxnQkFBZ0IsRUFBRSxNQUFNLGdDQUFnQyxDQUFDO0FBRWxFLFlBQVk7QUFDWixPQUFPLEVBQUUscUJBQXFCLEVBQUUsTUFBTSxxQ0FBcUMsQ0FBQztBQUM1RSxPQUFPLEVBQUUscUJBQXFCLEVBQUUsTUFBTSxxQ0FBcUMsQ0FBQztBQUM1RSxPQUFPLEVBQUUscUJBQXFCLEVBQUUsTUFBTSxxQ0FBcUMsQ0FBQztBQUM1RSxPQUFPLEVBQUUsd0JBQXdCLEVBQUUsTUFBTSx3Q0FBd0MsQ0FBQztBQUNsRixPQUFPLEVBQUUsc0JBQXNCLEVBQUUsTUFBTSxzQ0FBc0MsQ0FBQztBQUM5RSxPQUFPLEVBQUUsdUJBQXVCLEVBQUUsTUFBTSx1Q0FBdUMsQ0FBQztBQUVoRixvQkFBb0I7QUFDcEIsT0FBTyxFQUFFLHNCQUFzQixFQUFFLE1BQU0sVUFBVSxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLy8gVHlwZXNcbmV4cG9ydCB7XG4gIFNrZWxldG9uVGVtcGxhdGVOYW1lLFxuICBTa2VsZXRvblRlbXBsYXRlQ29uZmlnLFxuICBMb2FkaW5nU291cmNlLFxuICBTa2VsZXRvbkRpcmVjdGl2ZUNvbmZpZyxcbiAgTG9hZGluZ0RpcmVjdGl2ZUNvbmZpZyxcbiAgUmVnaXN0ZXJlZFNrZWxldG9uVGVtcGxhdGUsXG4gIFNrZWxldG9uQ29uZmlnLFxuICBTa2VsZXRvblRlbXBsYXRlQ29udGV4dCxcbiAgU2tlbGV0b25UZW1wbGF0ZUNvbXBvbmVudCxcbiAgREVGQVVMVF9TS0VMRVRPTl9DT05GSUcsXG59IGZyb20gJy4vdHlwZXMnO1xuXG4vLyBTZXJ2aWNlXG5leHBvcnQgeyBTa2VsZXRvblNlcnZpY2UgfSBmcm9tICcuL3NrZWxldG9uLnNlcnZpY2UnO1xuXG4vLyBEaXJlY3RpdmVzXG5leHBvcnQgeyBMb2FkaW5nRGlyZWN0aXZlIH0gZnJvbSAnLi9kaXJlY3RpdmVzL2xvYWRpbmcuZGlyZWN0aXZlJztcblxuLy8gVGVtcGxhdGVzXG5leHBvcnQgeyBMaXN0U2tlbGV0b25Db21wb25lbnQgfSBmcm9tICcuL3RlbXBsYXRlcy9saXN0LXNrZWxldG9uLmNvbXBvbmVudCc7XG5leHBvcnQgeyBHcmlkU2tlbGV0b25Db21wb25lbnQgfSBmcm9tICcuL3RlbXBsYXRlcy9ncmlkLXNrZWxldG9uLmNvbXBvbmVudCc7XG5leHBvcnQgeyBGb3JtU2tlbGV0b25Db21wb25lbnQgfSBmcm9tICcuL3RlbXBsYXRlcy9mb3JtLXNrZWxldG9uLmNvbXBvbmVudCc7XG5leHBvcnQgeyBQcm9maWxlU2tlbGV0b25Db21wb25lbnQgfSBmcm9tICcuL3RlbXBsYXRlcy9wcm9maWxlLXNrZWxldG9uLmNvbXBvbmVudCc7XG5leHBvcnQgeyBUYWJsZVNrZWxldG9uQ29tcG9uZW50IH0gZnJvbSAnLi90ZW1wbGF0ZXMvdGFibGUtc2tlbGV0b24uY29tcG9uZW50JztcbmV4cG9ydCB7IERldGFpbFNrZWxldG9uQ29tcG9uZW50IH0gZnJvbSAnLi90ZW1wbGF0ZXMvZGV0YWlsLXNrZWxldG9uLmNvbXBvbmVudCc7XG5cbi8vIENvbmZpZyAvIFByb3ZpZGVyXG5leHBvcnQgeyBwcm92aWRlVmFsdGVjaFNrZWxldG9uIH0gZnJvbSAnLi9jb25maWcnO1xuIl19
|
|
@@ -0,0 +1,198 @@
|
|
|
1
|
+
import { Injectable, signal, computed } from '@angular/core';
|
|
2
|
+
import { DEFAULT_SKELETON_CONFIG, } from './types';
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
/**
|
|
5
|
+
* Servicio para gestionar templates de skeleton y estados de carga globales.
|
|
6
|
+
*
|
|
7
|
+
* @example
|
|
8
|
+
* // En un componente
|
|
9
|
+
* skeleton = inject(SkeletonService);
|
|
10
|
+
*
|
|
11
|
+
* // Registrar template personalizado
|
|
12
|
+
* skeleton.registerTemplate('my-custom', MyCustomSkeletonComponent);
|
|
13
|
+
*
|
|
14
|
+
* // Obtener componente de template
|
|
15
|
+
* const component = skeleton.getTemplate('list');
|
|
16
|
+
*
|
|
17
|
+
* // Gestionar estados de carga globales
|
|
18
|
+
* skeleton.setLoadingState('dashboard', true);
|
|
19
|
+
* const isDashboardLoading = skeleton.loadingState('dashboard');
|
|
20
|
+
*/
|
|
21
|
+
export class SkeletonService {
|
|
22
|
+
constructor() {
|
|
23
|
+
this._config = signal(DEFAULT_SKELETON_CONFIG);
|
|
24
|
+
this._templates = signal(new Map());
|
|
25
|
+
this._loadingStates = signal(new Map());
|
|
26
|
+
this._initialized = false;
|
|
27
|
+
/** Configuracion actual (solo lectura) */
|
|
28
|
+
this.config = this._config.asReadonly();
|
|
29
|
+
/** Templates registrados (solo lectura) */
|
|
30
|
+
this.templates = this._templates.asReadonly();
|
|
31
|
+
/** Estado de carga global (cualquier estado registrado esta cargando) */
|
|
32
|
+
this.isAnyLoading = computed(() => {
|
|
33
|
+
const states = this._loadingStates();
|
|
34
|
+
return Array.from(states.values()).some((v) => v);
|
|
35
|
+
});
|
|
36
|
+
/** Cantidad de templates registrados */
|
|
37
|
+
this.templateCount = computed(() => this._templates().size);
|
|
38
|
+
}
|
|
39
|
+
/**
|
|
40
|
+
* Configura el servicio de skeleton.
|
|
41
|
+
* Llamado por provideValtechSkeleton().
|
|
42
|
+
*/
|
|
43
|
+
configure(config) {
|
|
44
|
+
if (this._initialized) {
|
|
45
|
+
console.warn('[SkeletonService] Service already configured. Ignoring reconfiguration.');
|
|
46
|
+
return;
|
|
47
|
+
}
|
|
48
|
+
this._config.set({ ...DEFAULT_SKELETON_CONFIG, ...config });
|
|
49
|
+
// Registrar templates personalizados de la configuracion
|
|
50
|
+
config.templates?.forEach((t) => {
|
|
51
|
+
this.registerTemplate(t.name, t.component, t.defaultConfig);
|
|
52
|
+
});
|
|
53
|
+
this._initialized = true;
|
|
54
|
+
}
|
|
55
|
+
/**
|
|
56
|
+
* Registra un template de skeleton personalizado.
|
|
57
|
+
*
|
|
58
|
+
* @param name Nombre unico del template
|
|
59
|
+
* @param component Componente a usar
|
|
60
|
+
* @param defaultConfig Configuracion por defecto opcional
|
|
61
|
+
*/
|
|
62
|
+
registerTemplate(name, component, defaultConfig) {
|
|
63
|
+
this._templates.update((map) => {
|
|
64
|
+
const newMap = new Map(map);
|
|
65
|
+
newMap.set(name, { name, component, defaultConfig });
|
|
66
|
+
return newMap;
|
|
67
|
+
});
|
|
68
|
+
}
|
|
69
|
+
/**
|
|
70
|
+
* Obtiene un template de skeleton registrado.
|
|
71
|
+
*
|
|
72
|
+
* @param name Nombre del template
|
|
73
|
+
* @returns Template registrado o undefined si no existe
|
|
74
|
+
*/
|
|
75
|
+
getTemplate(name) {
|
|
76
|
+
return this._templates().get(name);
|
|
77
|
+
}
|
|
78
|
+
/**
|
|
79
|
+
* Verifica si un template esta registrado.
|
|
80
|
+
*
|
|
81
|
+
* @param name Nombre del template
|
|
82
|
+
* @returns true si el template existe
|
|
83
|
+
*/
|
|
84
|
+
hasTemplate(name) {
|
|
85
|
+
return this._templates().has(name);
|
|
86
|
+
}
|
|
87
|
+
/**
|
|
88
|
+
* Obtiene todos los nombres de templates registrados.
|
|
89
|
+
*
|
|
90
|
+
* @returns Array de nombres de templates
|
|
91
|
+
*/
|
|
92
|
+
getTemplateNames() {
|
|
93
|
+
return Array.from(this._templates().keys());
|
|
94
|
+
}
|
|
95
|
+
/**
|
|
96
|
+
* Registra un estado de carga nombrado.
|
|
97
|
+
* Util para indicadores de carga globales.
|
|
98
|
+
*
|
|
99
|
+
* @param key Identificador unico del estado
|
|
100
|
+
* @param isLoading Estado de carga
|
|
101
|
+
*/
|
|
102
|
+
setLoadingState(key, isLoading) {
|
|
103
|
+
this._loadingStates.update((map) => {
|
|
104
|
+
const newMap = new Map(map);
|
|
105
|
+
if (isLoading) {
|
|
106
|
+
newMap.set(key, true);
|
|
107
|
+
}
|
|
108
|
+
else {
|
|
109
|
+
newMap.delete(key);
|
|
110
|
+
}
|
|
111
|
+
return newMap;
|
|
112
|
+
});
|
|
113
|
+
}
|
|
114
|
+
/**
|
|
115
|
+
* Obtiene el estado de carga para una clave especifica.
|
|
116
|
+
*
|
|
117
|
+
* @param key Identificador del estado
|
|
118
|
+
* @returns Estado de carga actual
|
|
119
|
+
*/
|
|
120
|
+
getLoadingState(key) {
|
|
121
|
+
return this._loadingStates().get(key) ?? false;
|
|
122
|
+
}
|
|
123
|
+
/**
|
|
124
|
+
* Crea un signal computado para un estado de carga especifico.
|
|
125
|
+
*
|
|
126
|
+
* @param key Identificador del estado
|
|
127
|
+
* @returns Signal reactivo del estado de carga
|
|
128
|
+
*/
|
|
129
|
+
loadingState(key) {
|
|
130
|
+
return computed(() => this._loadingStates().get(key) ?? false);
|
|
131
|
+
}
|
|
132
|
+
/**
|
|
133
|
+
* Obtiene todas las claves de estados de carga activos.
|
|
134
|
+
*
|
|
135
|
+
* @returns Array de claves con estado de carga activo
|
|
136
|
+
*/
|
|
137
|
+
getActiveLoadingKeys() {
|
|
138
|
+
const states = this._loadingStates();
|
|
139
|
+
return Array.from(states.entries())
|
|
140
|
+
.filter(([, isLoading]) => isLoading)
|
|
141
|
+
.map(([key]) => key);
|
|
142
|
+
}
|
|
143
|
+
/**
|
|
144
|
+
* Limpia todos los estados de carga.
|
|
145
|
+
*/
|
|
146
|
+
clearLoadingStates() {
|
|
147
|
+
this._loadingStates.set(new Map());
|
|
148
|
+
}
|
|
149
|
+
/**
|
|
150
|
+
* Limpia un estado de carga especifico.
|
|
151
|
+
*
|
|
152
|
+
* @param key Identificador del estado a limpiar
|
|
153
|
+
*/
|
|
154
|
+
clearLoadingState(key) {
|
|
155
|
+
this.setLoadingState(key, false);
|
|
156
|
+
}
|
|
157
|
+
/**
|
|
158
|
+
* Ejecuta una funcion async con tracking de estado de carga.
|
|
159
|
+
*
|
|
160
|
+
* @param key Identificador del estado
|
|
161
|
+
* @param fn Funcion async a ejecutar
|
|
162
|
+
* @returns Resultado de la funcion
|
|
163
|
+
*/
|
|
164
|
+
async withLoading(key, fn) {
|
|
165
|
+
this.setLoadingState(key, true);
|
|
166
|
+
try {
|
|
167
|
+
return await fn();
|
|
168
|
+
}
|
|
169
|
+
finally {
|
|
170
|
+
this.setLoadingState(key, false);
|
|
171
|
+
}
|
|
172
|
+
}
|
|
173
|
+
/**
|
|
174
|
+
* Obtiene la configuracion de animacion por defecto.
|
|
175
|
+
*/
|
|
176
|
+
get animated() {
|
|
177
|
+
return this._config().animated ?? true;
|
|
178
|
+
}
|
|
179
|
+
/**
|
|
180
|
+
* Obtiene el delay por defecto.
|
|
181
|
+
*/
|
|
182
|
+
get defaultDelay() {
|
|
183
|
+
return this._config().defaultDelay ?? 0;
|
|
184
|
+
}
|
|
185
|
+
/**
|
|
186
|
+
* Obtiene el tiempo minimo por defecto.
|
|
187
|
+
*/
|
|
188
|
+
get defaultMinTime() {
|
|
189
|
+
return this._config().defaultMinTime ?? 300;
|
|
190
|
+
}
|
|
191
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: SkeletonService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
192
|
+
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: SkeletonService, providedIn: 'root' }); }
|
|
193
|
+
}
|
|
194
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: SkeletonService, decorators: [{
|
|
195
|
+
type: Injectable,
|
|
196
|
+
args: [{ providedIn: 'root' }]
|
|
197
|
+
}] });
|
|
198
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"skeleton.service.js","sourceRoot":"","sources":["../../../../../../src/lib/services/skeleton/skeleton.service.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,EAAE,QAAQ,EAAQ,MAAM,eAAe,CAAC;AACnE,OAAO,EAKL,uBAAuB,GACxB,MAAM,SAAS,CAAC;;AAEjB;;;;;;;;;;;;;;;;GAgBG;AAEH,MAAM,OAAO,eAAe;IAD5B;QAEmB,YAAO,GAAG,MAAM,CAAiB,uBAAuB,CAAC,CAAC;QAC1D,eAAU,GAAG,MAAM,CAA0C,IAAI,GAAG,EAAE,CAAC,CAAC;QACxE,mBAAc,GAAG,MAAM,CAAuB,IAAI,GAAG,EAAE,CAAC,CAAC;QAClE,iBAAY,GAAG,KAAK,CAAC;QAE7B,0CAA0C;QACjC,WAAM,GAAG,IAAI,CAAC,OAAO,CAAC,UAAU,EAAE,CAAC;QAE5C,2CAA2C;QAClC,cAAS,GAAG,IAAI,CAAC,UAAU,CAAC,UAAU,EAAE,CAAC;QAElD,yEAAyE;QAChE,iBAAY,GAAG,QAAQ,CAAC,GAAG,EAAE;YACpC,MAAM,MAAM,GAAG,IAAI,CAAC,cAAc,EAAE,CAAC;YACrC,OAAO,KAAK,CAAC,IAAI,CAAC,MAAM,CAAC,MAAM,EAAE,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,CAAC;QACpD,CAAC,CAAC,CAAC;QAEH,wCAAwC;QAC/B,kBAAa,GAAG,QAAQ,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,UAAU,EAAE,CAAC,IAAI,CAAC,CAAC;KA6KjE;IA3KC;;;OAGG;IACH,SAAS,CAAC,MAAsB;QAC9B,IAAI,IAAI,CAAC,YAAY,EAAE,CAAC;YACtB,OAAO,CAAC,IAAI,CAAC,yEAAyE,CAAC,CAAC;YACxF,OAAO;QACT,CAAC;QAED,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,EAAE,GAAG,uBAAuB,EAAE,GAAG,MAAM,EAAE,CAAC,CAAC;QAE5D,yDAAyD;QACzD,MAAM,CAAC,SAAS,EAAE,OAAO,CAAC,CAAC,CAAC,EAAE,EAAE;YAC9B,IAAI,CAAC,gBAAgB,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC,CAAC,SAAS,EAAE,CAAC,CAAC,aAAa,CAAC,CAAC;QAC9D,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC;IAC3B,CAAC;IAED;;;;;;OAMG;IACH,gBAAgB,CACd,IAAY,EACZ,SAAwB,EACxB,aAAsC;QAEtC,IAAI,CAAC,UAAU,CAAC,MAAM,CAAC,CAAC,GAAG,EAAE,EAAE;YAC7B,MAAM,MAAM,GAAG,IAAI,GAAG,CAAC,GAAG,CAAC,CAAC;YAC5B,MAAM,CAAC,GAAG,CAAC,IAAI,EAAE,EAAE,IAAI,EAAE,SAAS,EAAE,aAAa,EAAE,CAAC,CAAC;YACrD,OAAO,MAAM,CAAC;QAChB,CAAC,CAAC,CAAC;IACL,CAAC;IAED;;;;;OAKG;IACH,WAAW,CAAC,IAAmC;QAC7C,OAAO,IAAI,CAAC,UAAU,EAAE,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;IACrC,CAAC;IAED;;;;;OAKG;IACH,WAAW,CAAC,IAAY;QACtB,OAAO,IAAI,CAAC,UAAU,EAAE,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;IACrC,CAAC;IAED;;;;OAIG;IACH,gBAAgB;QACd,OAAO,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,UAAU,EAAE,CAAC,IAAI,EAAE,CAAC,CAAC;IAC9C,CAAC;IAED;;;;;;OAMG;IACH,eAAe,CAAC,GAAW,EAAE,SAAkB;QAC7C,IAAI,CAAC,cAAc,CAAC,MAAM,CAAC,CAAC,GAAG,EAAE,EAAE;YACjC,MAAM,MAAM,GAAG,IAAI,GAAG,CAAC,GAAG,CAAC,CAAC;YAC5B,IAAI,SAAS,EAAE,CAAC;gBACd,MAAM,CAAC,GAAG,CAAC,GAAG,EAAE,IAAI,CAAC,CAAC;YACxB,CAAC;iBAAM,CAAC;gBACN,MAAM,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC;YACrB,CAAC;YACD,OAAO,MAAM,CAAC;QAChB,CAAC,CAAC,CAAC;IACL,CAAC;IAED;;;;;OAKG;IACH,eAAe,CAAC,GAAW;QACzB,OAAO,IAAI,CAAC,cAAc,EAAE,CAAC,GAAG,CAAC,GAAG,CAAC,IAAI,KAAK,CAAC;IACjD,CAAC;IAED;;;;;OAKG;IACH,YAAY,CAAC,GAAW;QACtB,OAAO,QAAQ,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,cAAc,EAAE,CAAC,GAAG,CAAC,GAAG,CAAC,IAAI,KAAK,CAAC,CAAC;IACjE,CAAC;IAED;;;;OAIG;IACH,oBAAoB;QAClB,MAAM,MAAM,GAAG,IAAI,CAAC,cAAc,EAAE,CAAC;QACrC,OAAO,KAAK,CAAC,IAAI,CAAC,MAAM,CAAC,OAAO,EAAE,CAAC;aAChC,MAAM,CAAC,CAAC,CAAC,EAAE,SAAS,CAAC,EAAE,EAAE,CAAC,SAAS,CAAC;aACpC,GAAG,CAAC,CAAC,CAAC,GAAG,CAAC,EAAE,EAAE,CAAC,GAAG,CAAC,CAAC;IACzB,CAAC;IAED;;OAEG;IACH,kBAAkB;QAChB,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,IAAI,GAAG,EAAE,CAAC,CAAC;IACrC,CAAC;IAED;;;;OAIG;IACH,iBAAiB,CAAC,GAAW;QAC3B,IAAI,CAAC,eAAe,CAAC,GAAG,EAAE,KAAK,CAAC,CAAC;IACnC,CAAC;IAED;;;;;;OAMG;IACH,KAAK,CAAC,WAAW,CAAI,GAAW,EAAE,EAAoB;QACpD,IAAI,CAAC,eAAe,CAAC,GAAG,EAAE,IAAI,CAAC,CAAC;QAChC,IAAI,CAAC;YACH,OAAO,MAAM,EAAE,EAAE,CAAC;QACpB,CAAC;gBAAS,CAAC;YACT,IAAI,CAAC,eAAe,CAAC,GAAG,EAAE,KAAK,CAAC,CAAC;QACnC,CAAC;IACH,CAAC;IAED;;OAEG;IACH,IAAI,QAAQ;QACV,OAAO,IAAI,CAAC,OAAO,EAAE,CAAC,QAAQ,IAAI,IAAI,CAAC;IACzC,CAAC;IAED;;OAEG;IACH,IAAI,YAAY;QACd,OAAO,IAAI,CAAC,OAAO,EAAE,CAAC,YAAY,IAAI,CAAC,CAAC;IAC1C,CAAC;IAED;;OAEG;IACH,IAAI,cAAc;QAChB,OAAO,IAAI,CAAC,OAAO,EAAE,CAAC,cAAc,IAAI,GAAG,CAAC;IAC9C,CAAC;+GA/LU,eAAe;mHAAf,eAAe,cADF,MAAM;;4FACnB,eAAe;kBAD3B,UAAU;mBAAC,EAAE,UAAU,EAAE,MAAM,EAAE","sourcesContent":["import { Injectable, signal, computed, Type } from '@angular/core';\nimport {\n  SkeletonTemplateName,\n  SkeletonTemplateConfig,\n  RegisteredSkeletonTemplate,\n  SkeletonConfig,\n  DEFAULT_SKELETON_CONFIG,\n} from './types';\n\n/**\n * Servicio para gestionar templates de skeleton y estados de carga globales.\n *\n * @example\n * // En un componente\n * skeleton = inject(SkeletonService);\n *\n * // Registrar template personalizado\n * skeleton.registerTemplate('my-custom', MyCustomSkeletonComponent);\n *\n * // Obtener componente de template\n * const component = skeleton.getTemplate('list');\n *\n * // Gestionar estados de carga globales\n * skeleton.setLoadingState('dashboard', true);\n * const isDashboardLoading = skeleton.loadingState('dashboard');\n */\n@Injectable({ providedIn: 'root' })\nexport class SkeletonService {\n  private readonly _config = signal<SkeletonConfig>(DEFAULT_SKELETON_CONFIG);\n  private readonly _templates = signal<Map<string, RegisteredSkeletonTemplate>>(new Map());\n  private readonly _loadingStates = signal<Map<string, boolean>>(new Map());\n  private _initialized = false;\n\n  /** Configuracion actual (solo lectura) */\n  readonly config = this._config.asReadonly();\n\n  /** Templates registrados (solo lectura) */\n  readonly templates = this._templates.asReadonly();\n\n  /** Estado de carga global (cualquier estado registrado esta cargando) */\n  readonly isAnyLoading = computed(() => {\n    const states = this._loadingStates();\n    return Array.from(states.values()).some((v) => v);\n  });\n\n  /** Cantidad de templates registrados */\n  readonly templateCount = computed(() => this._templates().size);\n\n  /**\n   * Configura el servicio de skeleton.\n   * Llamado por provideValtechSkeleton().\n   */\n  configure(config: SkeletonConfig): void {\n    if (this._initialized) {\n      console.warn('[SkeletonService] Service already configured. Ignoring reconfiguration.');\n      return;\n    }\n\n    this._config.set({ ...DEFAULT_SKELETON_CONFIG, ...config });\n\n    // Registrar templates personalizados de la configuracion\n    config.templates?.forEach((t) => {\n      this.registerTemplate(t.name, t.component, t.defaultConfig);\n    });\n\n    this._initialized = true;\n  }\n\n  /**\n   * Registra un template de skeleton personalizado.\n   *\n   * @param name Nombre unico del template\n   * @param component Componente a usar\n   * @param defaultConfig Configuracion por defecto opcional\n   */\n  registerTemplate(\n    name: string,\n    component: Type<unknown>,\n    defaultConfig?: SkeletonTemplateConfig\n  ): void {\n    this._templates.update((map) => {\n      const newMap = new Map(map);\n      newMap.set(name, { name, component, defaultConfig });\n      return newMap;\n    });\n  }\n\n  /**\n   * Obtiene un template de skeleton registrado.\n   *\n   * @param name Nombre del template\n   * @returns Template registrado o undefined si no existe\n   */\n  getTemplate(name: SkeletonTemplateName | string): RegisteredSkeletonTemplate | undefined {\n    return this._templates().get(name);\n  }\n\n  /**\n   * Verifica si un template esta registrado.\n   *\n   * @param name Nombre del template\n   * @returns true si el template existe\n   */\n  hasTemplate(name: string): boolean {\n    return this._templates().has(name);\n  }\n\n  /**\n   * Obtiene todos los nombres de templates registrados.\n   *\n   * @returns Array de nombres de templates\n   */\n  getTemplateNames(): string[] {\n    return Array.from(this._templates().keys());\n  }\n\n  /**\n   * Registra un estado de carga nombrado.\n   * Util para indicadores de carga globales.\n   *\n   * @param key Identificador unico del estado\n   * @param isLoading Estado de carga\n   */\n  setLoadingState(key: string, isLoading: boolean): void {\n    this._loadingStates.update((map) => {\n      const newMap = new Map(map);\n      if (isLoading) {\n        newMap.set(key, true);\n      } else {\n        newMap.delete(key);\n      }\n      return newMap;\n    });\n  }\n\n  /**\n   * Obtiene el estado de carga para una clave especifica.\n   *\n   * @param key Identificador del estado\n   * @returns Estado de carga actual\n   */\n  getLoadingState(key: string): boolean {\n    return this._loadingStates().get(key) ?? false;\n  }\n\n  /**\n   * Crea un signal computado para un estado de carga especifico.\n   *\n   * @param key Identificador del estado\n   * @returns Signal reactivo del estado de carga\n   */\n  loadingState(key: string) {\n    return computed(() => this._loadingStates().get(key) ?? false);\n  }\n\n  /**\n   * Obtiene todas las claves de estados de carga activos.\n   *\n   * @returns Array de claves con estado de carga activo\n   */\n  getActiveLoadingKeys(): string[] {\n    const states = this._loadingStates();\n    return Array.from(states.entries())\n      .filter(([, isLoading]) => isLoading)\n      .map(([key]) => key);\n  }\n\n  /**\n   * Limpia todos los estados de carga.\n   */\n  clearLoadingStates(): void {\n    this._loadingStates.set(new Map());\n  }\n\n  /**\n   * Limpia un estado de carga especifico.\n   *\n   * @param key Identificador del estado a limpiar\n   */\n  clearLoadingState(key: string): void {\n    this.setLoadingState(key, false);\n  }\n\n  /**\n   * Ejecuta una funcion async con tracking de estado de carga.\n   *\n   * @param key Identificador del estado\n   * @param fn Funcion async a ejecutar\n   * @returns Resultado de la funcion\n   */\n  async withLoading<T>(key: string, fn: () => Promise<T>): Promise<T> {\n    this.setLoadingState(key, true);\n    try {\n      return await fn();\n    } finally {\n      this.setLoadingState(key, false);\n    }\n  }\n\n  /**\n   * Obtiene la configuracion de animacion por defecto.\n   */\n  get animated(): boolean {\n    return this._config().animated ?? true;\n  }\n\n  /**\n   * Obtiene el delay por defecto.\n   */\n  get defaultDelay(): number {\n    return this._config().defaultDelay ?? 0;\n  }\n\n  /**\n   * Obtiene el tiempo minimo por defecto.\n   */\n  get defaultMinTime(): number {\n    return this._config().defaultMinTime ?? 300;\n  }\n}\n"]}
|