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,618 @@
|
|
|
1
|
+
import { Component, Input, Output, EventEmitter, signal, computed, ViewChild, inject, ChangeDetectorRef, } from '@angular/core';
|
|
2
|
+
import { CommonModule } from '@angular/common';
|
|
3
|
+
import { IonInfiniteScroll, IonInfiniteScrollContent, IonButton, IonSpinner, IonIcon, IonText, IonList, IonItem, } from '@ionic/angular/standalone';
|
|
4
|
+
import { firstValueFrom, isObservable } from 'rxjs';
|
|
5
|
+
import { DEFAULT_INFINITE_LIST_METADATA, } from './types';
|
|
6
|
+
import { RefresherComponent } from '../../molecules/refresher/refresher.component';
|
|
7
|
+
import { SkeletonService } from '../../../services/skeleton/skeleton.service';
|
|
8
|
+
import * as i0 from "@angular/core";
|
|
9
|
+
import * as i1 from "@angular/common";
|
|
10
|
+
/**
|
|
11
|
+
* Componente wrapper para listas con infinite scroll.
|
|
12
|
+
*
|
|
13
|
+
* @example
|
|
14
|
+
* <!-- Uso basico con data source -->
|
|
15
|
+
* <val-infinite-list
|
|
16
|
+
* [props]="{
|
|
17
|
+
* dataSource: { loadFn: loadUsers, trackBy: trackByUserId },
|
|
18
|
+
* itemTemplate: userTemplate,
|
|
19
|
+
* pageSize: 20,
|
|
20
|
+
* threshold: '150px'
|
|
21
|
+
* }"
|
|
22
|
+
* ></val-infinite-list>
|
|
23
|
+
*
|
|
24
|
+
* <ng-template #userTemplate let-user let-index="index">
|
|
25
|
+
* <val-card [props]="{ title: user.name, subtitle: user.email }">
|
|
26
|
+
* <p>{{ user.bio }}</p>
|
|
27
|
+
* </val-card>
|
|
28
|
+
* </ng-template>
|
|
29
|
+
*
|
|
30
|
+
* @example
|
|
31
|
+
* <!-- Con pull-to-refresh y estado vacio personalizado -->
|
|
32
|
+
* <val-infinite-list
|
|
33
|
+
* [props]="{
|
|
34
|
+
* dataSource: { loadFn: loadMessages },
|
|
35
|
+
* itemTemplate: messageTemplate,
|
|
36
|
+
* direction: 'both',
|
|
37
|
+
* enableRefresh: true,
|
|
38
|
+
* emptyState: {
|
|
39
|
+
* icon: 'chatbubbles-outline',
|
|
40
|
+
* title: 'Sin mensajes',
|
|
41
|
+
* message: 'Inicia una conversacion'
|
|
42
|
+
* },
|
|
43
|
+
* skeleton: { template: 'list', count: 5 }
|
|
44
|
+
* }"
|
|
45
|
+
* (refresh)="onRefresh($event)"
|
|
46
|
+
* ></val-infinite-list>
|
|
47
|
+
*/
|
|
48
|
+
export class InfiniteListComponent {
|
|
49
|
+
constructor() {
|
|
50
|
+
this.skeletonService = inject(SkeletonService);
|
|
51
|
+
this.cdr = inject(ChangeDetectorRef);
|
|
52
|
+
// === Events ===
|
|
53
|
+
this.loadMore = new EventEmitter();
|
|
54
|
+
this.refresh = new EventEmitter();
|
|
55
|
+
this.stateChange = new EventEmitter();
|
|
56
|
+
this.itemsChange = new EventEmitter();
|
|
57
|
+
this.errorOccurred = new EventEmitter();
|
|
58
|
+
// === Reactive State ===
|
|
59
|
+
this.items = signal([]);
|
|
60
|
+
this.state = signal('idle');
|
|
61
|
+
this.hasMoreBottom = signal(true);
|
|
62
|
+
this.hasMoreTop = signal(false);
|
|
63
|
+
this.error = signal(null);
|
|
64
|
+
this.isInitialLoad = signal(true);
|
|
65
|
+
this.currentPage = 0;
|
|
66
|
+
this.currentCursor = null;
|
|
67
|
+
/** Progreso de carga (0-1 si totalCount conocido) */
|
|
68
|
+
this.loadProgress = computed(() => {
|
|
69
|
+
if (!this.props?.dataSource?.totalCount)
|
|
70
|
+
return null;
|
|
71
|
+
return this.items().length / this.props.dataSource.totalCount;
|
|
72
|
+
});
|
|
73
|
+
/** Anuncio de estado para lectores de pantalla */
|
|
74
|
+
this.statusAnnouncement = computed(() => {
|
|
75
|
+
switch (this.state()) {
|
|
76
|
+
case 'loading':
|
|
77
|
+
return 'Cargando items...';
|
|
78
|
+
case 'error':
|
|
79
|
+
return `Error: ${this.error()?.message || 'Ocurrio un error'}`;
|
|
80
|
+
case 'complete':
|
|
81
|
+
return 'Todos los items han sido cargados';
|
|
82
|
+
default:
|
|
83
|
+
return '';
|
|
84
|
+
}
|
|
85
|
+
});
|
|
86
|
+
}
|
|
87
|
+
/** Props combinados con defaults */
|
|
88
|
+
get mergedProps() {
|
|
89
|
+
return { ...DEFAULT_INFINITE_LIST_METADATA, ...this.props };
|
|
90
|
+
}
|
|
91
|
+
/** Config del refresher */
|
|
92
|
+
get refresherConfig() {
|
|
93
|
+
return this.mergedProps.refreshConfig ?? {};
|
|
94
|
+
}
|
|
95
|
+
/** Componente de skeleton a usar */
|
|
96
|
+
get skeletonComponent() {
|
|
97
|
+
const templateName = this.mergedProps.skeleton?.template || 'list';
|
|
98
|
+
const template = this.skeletonService.getTemplate(templateName);
|
|
99
|
+
return template?.component ?? null;
|
|
100
|
+
}
|
|
101
|
+
/** Inputs para el skeleton */
|
|
102
|
+
get skeletonInputs() {
|
|
103
|
+
return {
|
|
104
|
+
config: {
|
|
105
|
+
count: this.mergedProps.skeleton?.count ?? 3,
|
|
106
|
+
animated: true,
|
|
107
|
+
...this.mergedProps.skeleton?.config,
|
|
108
|
+
},
|
|
109
|
+
};
|
|
110
|
+
}
|
|
111
|
+
ngOnInit() {
|
|
112
|
+
// Cargar items iniciales del dataSource si existen
|
|
113
|
+
if (this.props.dataSource.items?.length) {
|
|
114
|
+
this.items.set([...this.props.dataSource.items]);
|
|
115
|
+
this.isInitialLoad.set(false);
|
|
116
|
+
}
|
|
117
|
+
// Auto-cargar si esta habilitado
|
|
118
|
+
if (this.mergedProps.autoLoad && !this.items().length) {
|
|
119
|
+
this.loadInitial();
|
|
120
|
+
}
|
|
121
|
+
}
|
|
122
|
+
ngOnDestroy() {
|
|
123
|
+
// Cleanup
|
|
124
|
+
}
|
|
125
|
+
/** Funcion de tracking para ngFor */
|
|
126
|
+
trackByFn(index, item) {
|
|
127
|
+
if (this.props.dataSource.trackBy) {
|
|
128
|
+
return this.props.dataSource.trackBy(index, item);
|
|
129
|
+
}
|
|
130
|
+
return index;
|
|
131
|
+
}
|
|
132
|
+
/** Si debe mostrar el scroll inferior */
|
|
133
|
+
shouldShowBottomScroll() {
|
|
134
|
+
const dir = this.mergedProps.direction;
|
|
135
|
+
return (dir === 'bottom' || dir === 'both') && this.items().length > 0;
|
|
136
|
+
}
|
|
137
|
+
/** Carga inicial de datos */
|
|
138
|
+
async loadInitial() {
|
|
139
|
+
if (!this.props.dataSource.loadFn)
|
|
140
|
+
return;
|
|
141
|
+
this.state.set('loading');
|
|
142
|
+
this.stateChange.emit('loading');
|
|
143
|
+
this.error.set(null);
|
|
144
|
+
try {
|
|
145
|
+
const params = {
|
|
146
|
+
direction: 'bottom',
|
|
147
|
+
page: 0,
|
|
148
|
+
pageSize: this.mergedProps.pageSize ?? 20,
|
|
149
|
+
};
|
|
150
|
+
const result = await this.executeLoad(params);
|
|
151
|
+
this.items.set(result.items);
|
|
152
|
+
this.hasMoreBottom.set(result.hasMore);
|
|
153
|
+
this.currentPage = 1;
|
|
154
|
+
this.currentCursor = result.cursor;
|
|
155
|
+
this.isInitialLoad.set(false);
|
|
156
|
+
this.state.set('idle');
|
|
157
|
+
this.stateChange.emit('idle');
|
|
158
|
+
this.itemsChange.emit(this.items());
|
|
159
|
+
}
|
|
160
|
+
catch (err) {
|
|
161
|
+
this.handleError(err);
|
|
162
|
+
}
|
|
163
|
+
}
|
|
164
|
+
/** Cargar mas items en la parte inferior */
|
|
165
|
+
async loadBottom() {
|
|
166
|
+
if (!this.hasMoreBottom() || this.state() === 'loading')
|
|
167
|
+
return;
|
|
168
|
+
if (!this.props.dataSource.loadFn)
|
|
169
|
+
return;
|
|
170
|
+
this.state.set('loading');
|
|
171
|
+
this.stateChange.emit('loading');
|
|
172
|
+
try {
|
|
173
|
+
const params = {
|
|
174
|
+
direction: 'bottom',
|
|
175
|
+
page: this.currentPage,
|
|
176
|
+
pageSize: this.mergedProps.pageSize ?? 20,
|
|
177
|
+
cursor: this.currentCursor,
|
|
178
|
+
lastItem: this.items()[this.items().length - 1],
|
|
179
|
+
};
|
|
180
|
+
const result = await this.executeLoad(params);
|
|
181
|
+
this.items.update((current) => [...current, ...result.items]);
|
|
182
|
+
this.hasMoreBottom.set(result.hasMore);
|
|
183
|
+
this.currentPage++;
|
|
184
|
+
this.currentCursor = result.cursor;
|
|
185
|
+
this.state.set(result.hasMore ? 'idle' : 'complete');
|
|
186
|
+
this.stateChange.emit(result.hasMore ? 'idle' : 'complete');
|
|
187
|
+
this.itemsChange.emit(this.items());
|
|
188
|
+
}
|
|
189
|
+
catch (err) {
|
|
190
|
+
this.handleError(err);
|
|
191
|
+
}
|
|
192
|
+
}
|
|
193
|
+
/** Cargar mas items en la parte superior */
|
|
194
|
+
async loadTop() {
|
|
195
|
+
if (!this.hasMoreTop() || this.state() === 'loading')
|
|
196
|
+
return;
|
|
197
|
+
if (!this.props.dataSource.loadFn)
|
|
198
|
+
return;
|
|
199
|
+
this.state.set('loading');
|
|
200
|
+
try {
|
|
201
|
+
const params = {
|
|
202
|
+
direction: 'top',
|
|
203
|
+
page: 0,
|
|
204
|
+
pageSize: this.mergedProps.pageSize ?? 20,
|
|
205
|
+
firstItem: this.items()[0],
|
|
206
|
+
};
|
|
207
|
+
const result = await this.executeLoad(params);
|
|
208
|
+
this.items.update((current) => [...result.items, ...current]);
|
|
209
|
+
this.hasMoreTop.set(result.hasMore);
|
|
210
|
+
this.state.set('idle');
|
|
211
|
+
this.itemsChange.emit(this.items());
|
|
212
|
+
}
|
|
213
|
+
catch (err) {
|
|
214
|
+
this.handleError(err);
|
|
215
|
+
}
|
|
216
|
+
}
|
|
217
|
+
/** Refresh - recargar desde cero */
|
|
218
|
+
async refreshList() {
|
|
219
|
+
this.currentPage = 0;
|
|
220
|
+
this.currentCursor = null;
|
|
221
|
+
this.hasMoreBottom.set(true);
|
|
222
|
+
this.items.set([]);
|
|
223
|
+
await this.loadInitial();
|
|
224
|
+
}
|
|
225
|
+
/** Reintentar despues de error */
|
|
226
|
+
async retry() {
|
|
227
|
+
this.error.set(null);
|
|
228
|
+
if (this.items().length === 0) {
|
|
229
|
+
await this.loadInitial();
|
|
230
|
+
}
|
|
231
|
+
else {
|
|
232
|
+
await this.loadBottom();
|
|
233
|
+
}
|
|
234
|
+
}
|
|
235
|
+
/** Reset completo */
|
|
236
|
+
async reset() {
|
|
237
|
+
this.items.set([]);
|
|
238
|
+
this.currentPage = 0;
|
|
239
|
+
this.currentCursor = null;
|
|
240
|
+
this.hasMoreBottom.set(true);
|
|
241
|
+
this.hasMoreTop.set(false);
|
|
242
|
+
this.error.set(null);
|
|
243
|
+
this.isInitialLoad.set(true);
|
|
244
|
+
this.state.set('idle');
|
|
245
|
+
if (this.mergedProps.autoLoad) {
|
|
246
|
+
await this.loadInitial();
|
|
247
|
+
}
|
|
248
|
+
}
|
|
249
|
+
/** Agregar items al inicio */
|
|
250
|
+
prependItems(newItems) {
|
|
251
|
+
this.items.update((current) => [...newItems, ...current]);
|
|
252
|
+
this.itemsChange.emit(this.items());
|
|
253
|
+
}
|
|
254
|
+
/** Agregar items al final */
|
|
255
|
+
appendItems(newItems) {
|
|
256
|
+
this.items.update((current) => [...current, ...newItems]);
|
|
257
|
+
this.itemsChange.emit(this.items());
|
|
258
|
+
}
|
|
259
|
+
/** Actualizar un item por indice */
|
|
260
|
+
updateItem(index, item) {
|
|
261
|
+
this.items.update((current) => {
|
|
262
|
+
const updated = [...current];
|
|
263
|
+
updated[index] = item;
|
|
264
|
+
return updated;
|
|
265
|
+
});
|
|
266
|
+
this.itemsChange.emit(this.items());
|
|
267
|
+
}
|
|
268
|
+
/** Remover un item por indice */
|
|
269
|
+
removeItem(index) {
|
|
270
|
+
this.items.update((current) => current.filter((_, i) => i !== index));
|
|
271
|
+
this.itemsChange.emit(this.items());
|
|
272
|
+
}
|
|
273
|
+
/** Handler para evento de infinite scroll */
|
|
274
|
+
async onInfiniteScroll(event) {
|
|
275
|
+
await this.loadBottom();
|
|
276
|
+
event.target.complete();
|
|
277
|
+
}
|
|
278
|
+
/** Handler para evento de refresh */
|
|
279
|
+
async onRefreshTriggered(event) {
|
|
280
|
+
this.refresh.emit(event);
|
|
281
|
+
await this.refreshList();
|
|
282
|
+
event.complete();
|
|
283
|
+
}
|
|
284
|
+
async executeLoad(params) {
|
|
285
|
+
const loadFn = this.props.dataSource.loadFn;
|
|
286
|
+
const result = loadFn(params);
|
|
287
|
+
if (isObservable(result)) {
|
|
288
|
+
return await firstValueFrom(result);
|
|
289
|
+
}
|
|
290
|
+
return await result;
|
|
291
|
+
}
|
|
292
|
+
handleError(err) {
|
|
293
|
+
this.error.set(err);
|
|
294
|
+
this.state.set('error');
|
|
295
|
+
this.stateChange.emit('error');
|
|
296
|
+
this.errorOccurred.emit(err);
|
|
297
|
+
console.error('[InfiniteList] Error loading items:', err);
|
|
298
|
+
}
|
|
299
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: InfiniteListComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
300
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: InfiniteListComponent, isStandalone: true, selector: "val-infinite-list", inputs: { props: "props" }, outputs: { loadMore: "loadMore", refresh: "refresh", stateChange: "stateChange", itemsChange: "itemsChange", errorOccurred: "errorOccurred" }, viewQueries: [{ propertyName: "infiniteScroll", first: true, predicate: IonInfiniteScroll, descendants: true }], ngImport: i0, template: `
|
|
301
|
+
<!-- Pull to refresh wrapper -->
|
|
302
|
+
@if (mergedProps.enableRefresh) {
|
|
303
|
+
<val-refresher [props]="refresherConfig" (refresh)="onRefreshTriggered($event)">
|
|
304
|
+
<ng-container *ngTemplateOutlet="listContent"></ng-container>
|
|
305
|
+
</val-refresher>
|
|
306
|
+
} @else {
|
|
307
|
+
<ng-container *ngTemplateOutlet="listContent"></ng-container>
|
|
308
|
+
}
|
|
309
|
+
|
|
310
|
+
<!-- Main list content template -->
|
|
311
|
+
<ng-template #listContent>
|
|
312
|
+
<div
|
|
313
|
+
class="infinite-list-container"
|
|
314
|
+
[class]="mergedProps.cssClass"
|
|
315
|
+
[style.max-height]="mergedProps.maxHeight"
|
|
316
|
+
[style.overflow-y]="mergedProps.maxHeight ? 'auto' : 'visible'"
|
|
317
|
+
role="feed"
|
|
318
|
+
[attr.aria-busy]="state() === 'loading'"
|
|
319
|
+
[attr.aria-label]="mergedProps.ariaLabel"
|
|
320
|
+
[attr.aria-description]="mergedProps.ariaDescription"
|
|
321
|
+
>
|
|
322
|
+
<!-- Loading state (initial) -->
|
|
323
|
+
@if (state() === 'loading' && items().length === 0) {
|
|
324
|
+
<div class="infinite-list-skeleton">
|
|
325
|
+
@if (mergedProps.skeleton?.customTemplate) {
|
|
326
|
+
<ng-container *ngTemplateOutlet="mergedProps.skeleton.customTemplate"></ng-container>
|
|
327
|
+
} @else {
|
|
328
|
+
<ng-container *ngComponentOutlet="skeletonComponent; inputs: skeletonInputs"></ng-container>
|
|
329
|
+
}
|
|
330
|
+
</div>
|
|
331
|
+
}
|
|
332
|
+
|
|
333
|
+
<!-- Empty state -->
|
|
334
|
+
@if (state() === 'idle' && items().length === 0 && !isInitialLoad()) {
|
|
335
|
+
<div class="infinite-list-empty">
|
|
336
|
+
@if (mergedProps.emptyState?.template) {
|
|
337
|
+
<ng-container *ngTemplateOutlet="mergedProps.emptyState.template"></ng-container>
|
|
338
|
+
} @else {
|
|
339
|
+
@if (mergedProps.emptyState?.icon) {
|
|
340
|
+
<ion-icon [name]="mergedProps.emptyState.icon" size="large"></ion-icon>
|
|
341
|
+
}
|
|
342
|
+
@if (mergedProps.emptyState?.title) {
|
|
343
|
+
<h3>{{ mergedProps.emptyState.title }}</h3>
|
|
344
|
+
}
|
|
345
|
+
@if (mergedProps.emptyState?.message) {
|
|
346
|
+
<p>{{ mergedProps.emptyState.message }}</p>
|
|
347
|
+
}
|
|
348
|
+
}
|
|
349
|
+
</div>
|
|
350
|
+
}
|
|
351
|
+
|
|
352
|
+
<!-- Error state -->
|
|
353
|
+
@if (state() === 'error') {
|
|
354
|
+
<div class="infinite-list-error">
|
|
355
|
+
@if (mergedProps.errorState?.template) {
|
|
356
|
+
<ng-container
|
|
357
|
+
*ngTemplateOutlet="mergedProps.errorState.template; context: { error: error(), retry: retry.bind(this) }"
|
|
358
|
+
></ng-container>
|
|
359
|
+
} @else {
|
|
360
|
+
@if (mergedProps.errorState?.icon) {
|
|
361
|
+
<ion-icon [name]="mergedProps.errorState.icon" color="danger" size="large"></ion-icon>
|
|
362
|
+
} @else {
|
|
363
|
+
<ion-icon name="alert-circle-outline" color="danger" size="large"></ion-icon>
|
|
364
|
+
}
|
|
365
|
+
<h3>{{ mergedProps.errorState?.title || 'Error' }}</h3>
|
|
366
|
+
<p>{{ mergedProps.errorState?.message || error()?.message || 'Ocurrio un error' }}</p>
|
|
367
|
+
@if (mergedProps.errorState?.showRetry !== false) {
|
|
368
|
+
<ion-button fill="outline" (click)="retry()">
|
|
369
|
+
{{ mergedProps.errorState?.retryText || 'Reintentar' }}
|
|
370
|
+
</ion-button>
|
|
371
|
+
}
|
|
372
|
+
}
|
|
373
|
+
</div>
|
|
374
|
+
}
|
|
375
|
+
|
|
376
|
+
<!-- Items list -->
|
|
377
|
+
@if (items().length > 0) {
|
|
378
|
+
<div class="infinite-list-items" [class.with-dividers]="mergedProps.showDividers">
|
|
379
|
+
@for (item of items(); track trackByFn($index, item); let i = $index; let first = $first; let last = $last) {
|
|
380
|
+
<article
|
|
381
|
+
class="infinite-list-item"
|
|
382
|
+
[attr.aria-setsize]="mergedProps.dataSource.totalCount || null"
|
|
383
|
+
[attr.aria-posinset]="i + 1"
|
|
384
|
+
>
|
|
385
|
+
<ng-container
|
|
386
|
+
*ngTemplateOutlet="
|
|
387
|
+
mergedProps.itemTemplate;
|
|
388
|
+
context: { $implicit: item, index: i, first: first, last: last, count: items().length }
|
|
389
|
+
"
|
|
390
|
+
></ng-container>
|
|
391
|
+
</article>
|
|
392
|
+
}
|
|
393
|
+
</div>
|
|
394
|
+
}
|
|
395
|
+
|
|
396
|
+
<!-- Bottom infinite scroll -->
|
|
397
|
+
@if (shouldShowBottomScroll()) {
|
|
398
|
+
@if (mergedProps.useLoadMoreButton) {
|
|
399
|
+
<div class="infinite-list-load-more">
|
|
400
|
+
@if (hasMoreBottom()) {
|
|
401
|
+
<ion-button
|
|
402
|
+
fill="outline"
|
|
403
|
+
[color]="mergedProps.color"
|
|
404
|
+
[disabled]="state() === 'loading'"
|
|
405
|
+
(click)="loadBottom()"
|
|
406
|
+
>
|
|
407
|
+
@if (state() === 'loading') {
|
|
408
|
+
<ion-spinner [name]="mergedProps.spinnerType" slot="start"></ion-spinner>
|
|
409
|
+
}
|
|
410
|
+
{{ mergedProps.loadMoreText || 'Cargar mas' }}
|
|
411
|
+
</ion-button>
|
|
412
|
+
} @else {
|
|
413
|
+
<ion-text color="medium">{{ mergedProps.noMoreText || 'No hay mas items' }}</ion-text>
|
|
414
|
+
}
|
|
415
|
+
</div>
|
|
416
|
+
} @else {
|
|
417
|
+
<ion-infinite-scroll
|
|
418
|
+
[threshold]="mergedProps.threshold"
|
|
419
|
+
[disabled]="!hasMoreBottom()"
|
|
420
|
+
(ionInfinite)="onInfiniteScroll($event)"
|
|
421
|
+
>
|
|
422
|
+
<ion-infinite-scroll-content
|
|
423
|
+
[loadingSpinner]="mergedProps.spinnerType"
|
|
424
|
+
[loadingText]="state() === 'loading' ? 'Cargando...' : ''"
|
|
425
|
+
></ion-infinite-scroll-content>
|
|
426
|
+
</ion-infinite-scroll>
|
|
427
|
+
}
|
|
428
|
+
}
|
|
429
|
+
|
|
430
|
+
<!-- No more items indicator -->
|
|
431
|
+
@if (!hasMoreBottom() && items().length > 0 && !mergedProps.useLoadMoreButton) {
|
|
432
|
+
<div class="infinite-list-end">
|
|
433
|
+
<ion-text color="medium">{{ mergedProps.noMoreText || 'No hay mas items' }}</ion-text>
|
|
434
|
+
</div>
|
|
435
|
+
}
|
|
436
|
+
</div>
|
|
437
|
+
</ng-template>
|
|
438
|
+
|
|
439
|
+
<!-- Live region for accessibility announcements -->
|
|
440
|
+
<div class="sr-only" role="status" aria-live="polite" [attr.aria-atomic]="true">
|
|
441
|
+
{{ statusAnnouncement() }}
|
|
442
|
+
</div>
|
|
443
|
+
`, isInline: true, styles: [":host{display:block}.infinite-list-container{width:100%}.infinite-list-skeleton,.infinite-list-empty,.infinite-list-error{padding:24px 16px;display:flex;flex-direction:column;align-items:center;justify-content:center;text-align:center;gap:12px}.infinite-list-empty ion-icon,.infinite-list-error ion-icon{font-size:48px;opacity:.6}.infinite-list-empty h3,.infinite-list-error h3{margin:0;font-size:18px;font-weight:600}.infinite-list-empty p,.infinite-list-error p{margin:0;color:var(--ion-color-medium);font-size:14px}.infinite-list-items{&.with-dividers .infinite-list-item:not(:last-child){border-bottom:1px solid var(--ion-color-light-shade, #d7d8da)}}.infinite-list-load-more{display:flex;justify-content:center;padding:16px}.infinite-list-end{display:flex;justify-content:center;padding:16px;font-size:14px}.sr-only{position:absolute;width:1px;height:1px;padding:0;margin:-1px;overflow:hidden;clip:rect(0,0,0,0);white-space:nowrap;border:0}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgComponentOutlet, selector: "[ngComponentOutlet]", inputs: ["ngComponentOutlet", "ngComponentOutletInputs", "ngComponentOutletInjector", "ngComponentOutletContent", "ngComponentOutletNgModule", "ngComponentOutletNgModuleFactory"] }, { kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "component", type: IonInfiniteScroll, selector: "ion-infinite-scroll", inputs: ["disabled", "position", "threshold"] }, { kind: "component", type: IonInfiniteScrollContent, selector: "ion-infinite-scroll-content", inputs: ["loadingSpinner", "loadingText"] }, { kind: "component", type: IonButton, selector: "ion-button", inputs: ["buttonType", "color", "disabled", "download", "expand", "fill", "form", "href", "mode", "rel", "routerAnimation", "routerDirection", "shape", "size", "strong", "target", "type"] }, { kind: "component", type: IonSpinner, selector: "ion-spinner", inputs: ["color", "duration", "name", "paused"] }, { kind: "component", type: IonIcon, selector: "ion-icon", inputs: ["color", "flipRtl", "icon", "ios", "lazy", "md", "mode", "name", "sanitize", "size", "src"] }, { kind: "component", type: IonText, selector: "ion-text", inputs: ["color", "mode"] }, { kind: "component", type: RefresherComponent, selector: "val-refresher", inputs: ["props"], outputs: ["refresh", "pullProgressChange", "stateChange"] }] }); }
|
|
444
|
+
}
|
|
445
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: InfiniteListComponent, decorators: [{
|
|
446
|
+
type: Component,
|
|
447
|
+
args: [{ selector: 'val-infinite-list', standalone: true, imports: [
|
|
448
|
+
CommonModule,
|
|
449
|
+
IonInfiniteScroll,
|
|
450
|
+
IonInfiniteScrollContent,
|
|
451
|
+
IonButton,
|
|
452
|
+
IonSpinner,
|
|
453
|
+
IonIcon,
|
|
454
|
+
IonText,
|
|
455
|
+
IonList,
|
|
456
|
+
IonItem,
|
|
457
|
+
RefresherComponent,
|
|
458
|
+
], template: `
|
|
459
|
+
<!-- Pull to refresh wrapper -->
|
|
460
|
+
@if (mergedProps.enableRefresh) {
|
|
461
|
+
<val-refresher [props]="refresherConfig" (refresh)="onRefreshTriggered($event)">
|
|
462
|
+
<ng-container *ngTemplateOutlet="listContent"></ng-container>
|
|
463
|
+
</val-refresher>
|
|
464
|
+
} @else {
|
|
465
|
+
<ng-container *ngTemplateOutlet="listContent"></ng-container>
|
|
466
|
+
}
|
|
467
|
+
|
|
468
|
+
<!-- Main list content template -->
|
|
469
|
+
<ng-template #listContent>
|
|
470
|
+
<div
|
|
471
|
+
class="infinite-list-container"
|
|
472
|
+
[class]="mergedProps.cssClass"
|
|
473
|
+
[style.max-height]="mergedProps.maxHeight"
|
|
474
|
+
[style.overflow-y]="mergedProps.maxHeight ? 'auto' : 'visible'"
|
|
475
|
+
role="feed"
|
|
476
|
+
[attr.aria-busy]="state() === 'loading'"
|
|
477
|
+
[attr.aria-label]="mergedProps.ariaLabel"
|
|
478
|
+
[attr.aria-description]="mergedProps.ariaDescription"
|
|
479
|
+
>
|
|
480
|
+
<!-- Loading state (initial) -->
|
|
481
|
+
@if (state() === 'loading' && items().length === 0) {
|
|
482
|
+
<div class="infinite-list-skeleton">
|
|
483
|
+
@if (mergedProps.skeleton?.customTemplate) {
|
|
484
|
+
<ng-container *ngTemplateOutlet="mergedProps.skeleton.customTemplate"></ng-container>
|
|
485
|
+
} @else {
|
|
486
|
+
<ng-container *ngComponentOutlet="skeletonComponent; inputs: skeletonInputs"></ng-container>
|
|
487
|
+
}
|
|
488
|
+
</div>
|
|
489
|
+
}
|
|
490
|
+
|
|
491
|
+
<!-- Empty state -->
|
|
492
|
+
@if (state() === 'idle' && items().length === 0 && !isInitialLoad()) {
|
|
493
|
+
<div class="infinite-list-empty">
|
|
494
|
+
@if (mergedProps.emptyState?.template) {
|
|
495
|
+
<ng-container *ngTemplateOutlet="mergedProps.emptyState.template"></ng-container>
|
|
496
|
+
} @else {
|
|
497
|
+
@if (mergedProps.emptyState?.icon) {
|
|
498
|
+
<ion-icon [name]="mergedProps.emptyState.icon" size="large"></ion-icon>
|
|
499
|
+
}
|
|
500
|
+
@if (mergedProps.emptyState?.title) {
|
|
501
|
+
<h3>{{ mergedProps.emptyState.title }}</h3>
|
|
502
|
+
}
|
|
503
|
+
@if (mergedProps.emptyState?.message) {
|
|
504
|
+
<p>{{ mergedProps.emptyState.message }}</p>
|
|
505
|
+
}
|
|
506
|
+
}
|
|
507
|
+
</div>
|
|
508
|
+
}
|
|
509
|
+
|
|
510
|
+
<!-- Error state -->
|
|
511
|
+
@if (state() === 'error') {
|
|
512
|
+
<div class="infinite-list-error">
|
|
513
|
+
@if (mergedProps.errorState?.template) {
|
|
514
|
+
<ng-container
|
|
515
|
+
*ngTemplateOutlet="mergedProps.errorState.template; context: { error: error(), retry: retry.bind(this) }"
|
|
516
|
+
></ng-container>
|
|
517
|
+
} @else {
|
|
518
|
+
@if (mergedProps.errorState?.icon) {
|
|
519
|
+
<ion-icon [name]="mergedProps.errorState.icon" color="danger" size="large"></ion-icon>
|
|
520
|
+
} @else {
|
|
521
|
+
<ion-icon name="alert-circle-outline" color="danger" size="large"></ion-icon>
|
|
522
|
+
}
|
|
523
|
+
<h3>{{ mergedProps.errorState?.title || 'Error' }}</h3>
|
|
524
|
+
<p>{{ mergedProps.errorState?.message || error()?.message || 'Ocurrio un error' }}</p>
|
|
525
|
+
@if (mergedProps.errorState?.showRetry !== false) {
|
|
526
|
+
<ion-button fill="outline" (click)="retry()">
|
|
527
|
+
{{ mergedProps.errorState?.retryText || 'Reintentar' }}
|
|
528
|
+
</ion-button>
|
|
529
|
+
}
|
|
530
|
+
}
|
|
531
|
+
</div>
|
|
532
|
+
}
|
|
533
|
+
|
|
534
|
+
<!-- Items list -->
|
|
535
|
+
@if (items().length > 0) {
|
|
536
|
+
<div class="infinite-list-items" [class.with-dividers]="mergedProps.showDividers">
|
|
537
|
+
@for (item of items(); track trackByFn($index, item); let i = $index; let first = $first; let last = $last) {
|
|
538
|
+
<article
|
|
539
|
+
class="infinite-list-item"
|
|
540
|
+
[attr.aria-setsize]="mergedProps.dataSource.totalCount || null"
|
|
541
|
+
[attr.aria-posinset]="i + 1"
|
|
542
|
+
>
|
|
543
|
+
<ng-container
|
|
544
|
+
*ngTemplateOutlet="
|
|
545
|
+
mergedProps.itemTemplate;
|
|
546
|
+
context: { $implicit: item, index: i, first: first, last: last, count: items().length }
|
|
547
|
+
"
|
|
548
|
+
></ng-container>
|
|
549
|
+
</article>
|
|
550
|
+
}
|
|
551
|
+
</div>
|
|
552
|
+
}
|
|
553
|
+
|
|
554
|
+
<!-- Bottom infinite scroll -->
|
|
555
|
+
@if (shouldShowBottomScroll()) {
|
|
556
|
+
@if (mergedProps.useLoadMoreButton) {
|
|
557
|
+
<div class="infinite-list-load-more">
|
|
558
|
+
@if (hasMoreBottom()) {
|
|
559
|
+
<ion-button
|
|
560
|
+
fill="outline"
|
|
561
|
+
[color]="mergedProps.color"
|
|
562
|
+
[disabled]="state() === 'loading'"
|
|
563
|
+
(click)="loadBottom()"
|
|
564
|
+
>
|
|
565
|
+
@if (state() === 'loading') {
|
|
566
|
+
<ion-spinner [name]="mergedProps.spinnerType" slot="start"></ion-spinner>
|
|
567
|
+
}
|
|
568
|
+
{{ mergedProps.loadMoreText || 'Cargar mas' }}
|
|
569
|
+
</ion-button>
|
|
570
|
+
} @else {
|
|
571
|
+
<ion-text color="medium">{{ mergedProps.noMoreText || 'No hay mas items' }}</ion-text>
|
|
572
|
+
}
|
|
573
|
+
</div>
|
|
574
|
+
} @else {
|
|
575
|
+
<ion-infinite-scroll
|
|
576
|
+
[threshold]="mergedProps.threshold"
|
|
577
|
+
[disabled]="!hasMoreBottom()"
|
|
578
|
+
(ionInfinite)="onInfiniteScroll($event)"
|
|
579
|
+
>
|
|
580
|
+
<ion-infinite-scroll-content
|
|
581
|
+
[loadingSpinner]="mergedProps.spinnerType"
|
|
582
|
+
[loadingText]="state() === 'loading' ? 'Cargando...' : ''"
|
|
583
|
+
></ion-infinite-scroll-content>
|
|
584
|
+
</ion-infinite-scroll>
|
|
585
|
+
}
|
|
586
|
+
}
|
|
587
|
+
|
|
588
|
+
<!-- No more items indicator -->
|
|
589
|
+
@if (!hasMoreBottom() && items().length > 0 && !mergedProps.useLoadMoreButton) {
|
|
590
|
+
<div class="infinite-list-end">
|
|
591
|
+
<ion-text color="medium">{{ mergedProps.noMoreText || 'No hay mas items' }}</ion-text>
|
|
592
|
+
</div>
|
|
593
|
+
}
|
|
594
|
+
</div>
|
|
595
|
+
</ng-template>
|
|
596
|
+
|
|
597
|
+
<!-- Live region for accessibility announcements -->
|
|
598
|
+
<div class="sr-only" role="status" aria-live="polite" [attr.aria-atomic]="true">
|
|
599
|
+
{{ statusAnnouncement() }}
|
|
600
|
+
</div>
|
|
601
|
+
`, styles: [":host{display:block}.infinite-list-container{width:100%}.infinite-list-skeleton,.infinite-list-empty,.infinite-list-error{padding:24px 16px;display:flex;flex-direction:column;align-items:center;justify-content:center;text-align:center;gap:12px}.infinite-list-empty ion-icon,.infinite-list-error ion-icon{font-size:48px;opacity:.6}.infinite-list-empty h3,.infinite-list-error h3{margin:0;font-size:18px;font-weight:600}.infinite-list-empty p,.infinite-list-error p{margin:0;color:var(--ion-color-medium);font-size:14px}.infinite-list-items{&.with-dividers .infinite-list-item:not(:last-child){border-bottom:1px solid var(--ion-color-light-shade, #d7d8da)}}.infinite-list-load-more{display:flex;justify-content:center;padding:16px}.infinite-list-end{display:flex;justify-content:center;padding:16px;font-size:14px}.sr-only{position:absolute;width:1px;height:1px;padding:0;margin:-1px;overflow:hidden;clip:rect(0,0,0,0);white-space:nowrap;border:0}\n"] }]
|
|
602
|
+
}], propDecorators: { infiniteScroll: [{
|
|
603
|
+
type: ViewChild,
|
|
604
|
+
args: [IonInfiniteScroll]
|
|
605
|
+
}], props: [{
|
|
606
|
+
type: Input
|
|
607
|
+
}], loadMore: [{
|
|
608
|
+
type: Output
|
|
609
|
+
}], refresh: [{
|
|
610
|
+
type: Output
|
|
611
|
+
}], stateChange: [{
|
|
612
|
+
type: Output
|
|
613
|
+
}], itemsChange: [{
|
|
614
|
+
type: Output
|
|
615
|
+
}], errorOccurred: [{
|
|
616
|
+
type: Output
|
|
617
|
+
}] } });
|
|
618
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"infinite-list.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/organisms/infinite-list/infinite-list.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  Input,\n  Output,\n  EventEmitter,\n  signal,\n  computed,\n  OnInit,\n  OnDestroy,\n  ViewChild,\n  inject,\n  ChangeDetectorRef,\n} from '@angular/core';\nimport { CommonModule } from '@angular/common';\nimport {\n  IonInfiniteScroll,\n  IonInfiniteScrollContent,\n  IonButton,\n  IonSpinner,\n  IonIcon,\n  IonText,\n  IonList,\n  IonItem,\n} from '@ionic/angular/standalone';\nimport { firstValueFrom, isObservable } from 'rxjs';\nimport {\n  InfiniteListMetadata,\n  InfiniteListState,\n  LoadMoreEvent,\n  LoadParams,\n  LoadResult,\n  DEFAULT_INFINITE_LIST_METADATA,\n} from './types';\nimport { RefreshEvent, RefresherMetadata } from '../../molecules/refresher/types';\nimport { RefresherComponent } from '../../molecules/refresher/refresher.component';\nimport { SkeletonService } from '../../../services/skeleton/skeleton.service';\nimport { SkeletonTemplateComponent } from '../../../services/skeleton/types';\n\n/**\n * Componente wrapper para listas con infinite scroll.\n *\n * @example\n * <!-- Uso basico con data source -->\n * <val-infinite-list\n *   [props]=\"{\n *     dataSource: { loadFn: loadUsers, trackBy: trackByUserId },\n *     itemTemplate: userTemplate,\n *     pageSize: 20,\n *     threshold: '150px'\n *   }\"\n * ></val-infinite-list>\n *\n * <ng-template #userTemplate let-user let-index=\"index\">\n *   <val-card [props]=\"{ title: user.name, subtitle: user.email }\">\n *     <p>{{ user.bio }}</p>\n *   </val-card>\n * </ng-template>\n *\n * @example\n * <!-- Con pull-to-refresh y estado vacio personalizado -->\n * <val-infinite-list\n *   [props]=\"{\n *     dataSource: { loadFn: loadMessages },\n *     itemTemplate: messageTemplate,\n *     direction: 'both',\n *     enableRefresh: true,\n *     emptyState: {\n *       icon: 'chatbubbles-outline',\n *       title: 'Sin mensajes',\n *       message: 'Inicia una conversacion'\n *     },\n *     skeleton: { template: 'list', count: 5 }\n *   }\"\n *   (refresh)=\"onRefresh($event)\"\n * ></val-infinite-list>\n */\n@Component({\n  selector: 'val-infinite-list',\n  standalone: true,\n  imports: [\n    CommonModule,\n    IonInfiniteScroll,\n    IonInfiniteScrollContent,\n    IonButton,\n    IonSpinner,\n    IonIcon,\n    IonText,\n    IonList,\n    IonItem,\n    RefresherComponent,\n  ],\n  template: `\n    <!-- Pull to refresh wrapper -->\n    @if (mergedProps.enableRefresh) {\n      <val-refresher [props]=\"refresherConfig\" (refresh)=\"onRefreshTriggered($event)\">\n        <ng-container *ngTemplateOutlet=\"listContent\"></ng-container>\n      </val-refresher>\n    } @else {\n      <ng-container *ngTemplateOutlet=\"listContent\"></ng-container>\n    }\n\n    <!-- Main list content template -->\n    <ng-template #listContent>\n      <div\n        class=\"infinite-list-container\"\n        [class]=\"mergedProps.cssClass\"\n        [style.max-height]=\"mergedProps.maxHeight\"\n        [style.overflow-y]=\"mergedProps.maxHeight ? 'auto' : 'visible'\"\n        role=\"feed\"\n        [attr.aria-busy]=\"state() === 'loading'\"\n        [attr.aria-label]=\"mergedProps.ariaLabel\"\n        [attr.aria-description]=\"mergedProps.ariaDescription\"\n      >\n        <!-- Loading state (initial) -->\n        @if (state() === 'loading' && items().length === 0) {\n          <div class=\"infinite-list-skeleton\">\n            @if (mergedProps.skeleton?.customTemplate) {\n              <ng-container *ngTemplateOutlet=\"mergedProps.skeleton.customTemplate\"></ng-container>\n            } @else {\n              <ng-container *ngComponentOutlet=\"skeletonComponent; inputs: skeletonInputs\"></ng-container>\n            }\n          </div>\n        }\n\n        <!-- Empty state -->\n        @if (state() === 'idle' && items().length === 0 && !isInitialLoad()) {\n          <div class=\"infinite-list-empty\">\n            @if (mergedProps.emptyState?.template) {\n              <ng-container *ngTemplateOutlet=\"mergedProps.emptyState.template\"></ng-container>\n            } @else {\n              @if (mergedProps.emptyState?.icon) {\n                <ion-icon [name]=\"mergedProps.emptyState.icon\" size=\"large\"></ion-icon>\n              }\n              @if (mergedProps.emptyState?.title) {\n                <h3>{{ mergedProps.emptyState.title }}</h3>\n              }\n              @if (mergedProps.emptyState?.message) {\n                <p>{{ mergedProps.emptyState.message }}</p>\n              }\n            }\n          </div>\n        }\n\n        <!-- Error state -->\n        @if (state() === 'error') {\n          <div class=\"infinite-list-error\">\n            @if (mergedProps.errorState?.template) {\n              <ng-container\n                *ngTemplateOutlet=\"mergedProps.errorState.template; context: { error: error(), retry: retry.bind(this) }\"\n              ></ng-container>\n            } @else {\n              @if (mergedProps.errorState?.icon) {\n                <ion-icon [name]=\"mergedProps.errorState.icon\" color=\"danger\" size=\"large\"></ion-icon>\n              } @else {\n                <ion-icon name=\"alert-circle-outline\" color=\"danger\" size=\"large\"></ion-icon>\n              }\n              <h3>{{ mergedProps.errorState?.title || 'Error' }}</h3>\n              <p>{{ mergedProps.errorState?.message || error()?.message || 'Ocurrio un error' }}</p>\n              @if (mergedProps.errorState?.showRetry !== false) {\n                <ion-button fill=\"outline\" (click)=\"retry()\">\n                  {{ mergedProps.errorState?.retryText || 'Reintentar' }}\n                </ion-button>\n              }\n            }\n          </div>\n        }\n\n        <!-- Items list -->\n        @if (items().length > 0) {\n          <div class=\"infinite-list-items\" [class.with-dividers]=\"mergedProps.showDividers\">\n            @for (item of items(); track trackByFn($index, item); let i = $index; let first = $first; let last = $last) {\n              <article\n                class=\"infinite-list-item\"\n                [attr.aria-setsize]=\"mergedProps.dataSource.totalCount || null\"\n                [attr.aria-posinset]=\"i + 1\"\n              >\n                <ng-container\n                  *ngTemplateOutlet=\"\n                    mergedProps.itemTemplate;\n                    context: { $implicit: item, index: i, first: first, last: last, count: items().length }\n                  \"\n                ></ng-container>\n              </article>\n            }\n          </div>\n        }\n\n        <!-- Bottom infinite scroll -->\n        @if (shouldShowBottomScroll()) {\n          @if (mergedProps.useLoadMoreButton) {\n            <div class=\"infinite-list-load-more\">\n              @if (hasMoreBottom()) {\n                <ion-button\n                  fill=\"outline\"\n                  [color]=\"mergedProps.color\"\n                  [disabled]=\"state() === 'loading'\"\n                  (click)=\"loadBottom()\"\n                >\n                  @if (state() === 'loading') {\n                    <ion-spinner [name]=\"mergedProps.spinnerType\" slot=\"start\"></ion-spinner>\n                  }\n                  {{ mergedProps.loadMoreText || 'Cargar mas' }}\n                </ion-button>\n              } @else {\n                <ion-text color=\"medium\">{{ mergedProps.noMoreText || 'No hay mas items' }}</ion-text>\n              }\n            </div>\n          } @else {\n            <ion-infinite-scroll\n              [threshold]=\"mergedProps.threshold\"\n              [disabled]=\"!hasMoreBottom()\"\n              (ionInfinite)=\"onInfiniteScroll($event)\"\n            >\n              <ion-infinite-scroll-content\n                [loadingSpinner]=\"mergedProps.spinnerType\"\n                [loadingText]=\"state() === 'loading' ? 'Cargando...' : ''\"\n              ></ion-infinite-scroll-content>\n            </ion-infinite-scroll>\n          }\n        }\n\n        <!-- No more items indicator -->\n        @if (!hasMoreBottom() && items().length > 0 && !mergedProps.useLoadMoreButton) {\n          <div class=\"infinite-list-end\">\n            <ion-text color=\"medium\">{{ mergedProps.noMoreText || 'No hay mas items' }}</ion-text>\n          </div>\n        }\n      </div>\n    </ng-template>\n\n    <!-- Live region for accessibility announcements -->\n    <div class=\"sr-only\" role=\"status\" aria-live=\"polite\" [attr.aria-atomic]=\"true\">\n      {{ statusAnnouncement() }}\n    </div>\n  `,\n  styles: [\n    `\n      :host {\n        display: block;\n      }\n\n      .infinite-list-container {\n        width: 100%;\n      }\n\n      .infinite-list-skeleton,\n      .infinite-list-empty,\n      .infinite-list-error {\n        padding: 24px 16px;\n        display: flex;\n        flex-direction: column;\n        align-items: center;\n        justify-content: center;\n        text-align: center;\n        gap: 12px;\n      }\n\n      .infinite-list-empty ion-icon,\n      .infinite-list-error ion-icon {\n        font-size: 48px;\n        opacity: 0.6;\n      }\n\n      .infinite-list-empty h3,\n      .infinite-list-error h3 {\n        margin: 0;\n        font-size: 18px;\n        font-weight: 600;\n      }\n\n      .infinite-list-empty p,\n      .infinite-list-error p {\n        margin: 0;\n        color: var(--ion-color-medium);\n        font-size: 14px;\n      }\n\n      .infinite-list-items {\n        &.with-dividers .infinite-list-item:not(:last-child) {\n          border-bottom: 1px solid var(--ion-color-light-shade, #d7d8da);\n        }\n      }\n\n      .infinite-list-load-more {\n        display: flex;\n        justify-content: center;\n        padding: 16px;\n      }\n\n      .infinite-list-end {\n        display: flex;\n        justify-content: center;\n        padding: 16px;\n        font-size: 14px;\n      }\n\n      .sr-only {\n        position: absolute;\n        width: 1px;\n        height: 1px;\n        padding: 0;\n        margin: -1px;\n        overflow: hidden;\n        clip: rect(0, 0, 0, 0);\n        white-space: nowrap;\n        border: 0;\n      }\n    `,\n  ],\n})\nexport class InfiniteListComponent<T = unknown> implements OnInit, OnDestroy {\n  private readonly skeletonService = inject(SkeletonService);\n  private readonly cdr = inject(ChangeDetectorRef);\n\n  @ViewChild(IonInfiniteScroll) infiniteScroll?: IonInfiniteScroll;\n\n  /** Configuracion del componente */\n  @Input() props!: InfiniteListMetadata<T>;\n\n  // === Events ===\n  @Output() loadMore = new EventEmitter<LoadMoreEvent>();\n  @Output() refresh = new EventEmitter<RefreshEvent>();\n  @Output() stateChange = new EventEmitter<InfiniteListState>();\n  @Output() itemsChange = new EventEmitter<T[]>();\n  @Output() errorOccurred = new EventEmitter<Error>();\n\n  // === Reactive State ===\n  readonly items = signal<T[]>([]);\n  readonly state = signal<InfiniteListState>('idle');\n  readonly hasMoreBottom = signal<boolean>(true);\n  readonly hasMoreTop = signal<boolean>(false);\n  readonly error = signal<Error | null>(null);\n  readonly isInitialLoad = signal<boolean>(true);\n\n  private currentPage = 0;\n  private currentCursor: unknown = null;\n\n  /** Progreso de carga (0-1 si totalCount conocido) */\n  readonly loadProgress = computed(() => {\n    if (!this.props?.dataSource?.totalCount) return null;\n    return this.items().length / this.props.dataSource.totalCount;\n  });\n\n  /** Props combinados con defaults */\n  get mergedProps(): InfiniteListMetadata<T> {\n    return { ...DEFAULT_INFINITE_LIST_METADATA, ...this.props } as InfiniteListMetadata<T>;\n  }\n\n  /** Config del refresher */\n  get refresherConfig(): RefresherMetadata {\n    return this.mergedProps.refreshConfig ?? {};\n  }\n\n  /** Componente de skeleton a usar */\n  get skeletonComponent() {\n    const templateName = this.mergedProps.skeleton?.template || 'list';\n    const template = this.skeletonService.getTemplate(templateName);\n    return template?.component ?? null;\n  }\n\n  /** Inputs para el skeleton */\n  get skeletonInputs(): { config: unknown } {\n    return {\n      config: {\n        count: this.mergedProps.skeleton?.count ?? 3,\n        animated: true,\n        ...this.mergedProps.skeleton?.config,\n      },\n    };\n  }\n\n  /** Anuncio de estado para lectores de pantalla */\n  readonly statusAnnouncement = computed(() => {\n    switch (this.state()) {\n      case 'loading':\n        return 'Cargando items...';\n      case 'error':\n        return `Error: ${this.error()?.message || 'Ocurrio un error'}`;\n      case 'complete':\n        return 'Todos los items han sido cargados';\n      default:\n        return '';\n    }\n  });\n\n  ngOnInit(): void {\n    // Cargar items iniciales del dataSource si existen\n    if (this.props.dataSource.items?.length) {\n      this.items.set([...this.props.dataSource.items]);\n      this.isInitialLoad.set(false);\n    }\n\n    // Auto-cargar si esta habilitado\n    if (this.mergedProps.autoLoad && !this.items().length) {\n      this.loadInitial();\n    }\n  }\n\n  ngOnDestroy(): void {\n    // Cleanup\n  }\n\n  /** Funcion de tracking para ngFor */\n  trackByFn(index: number, item: T): unknown {\n    if (this.props.dataSource.trackBy) {\n      return this.props.dataSource.trackBy(index, item);\n    }\n    return index;\n  }\n\n  /** Si debe mostrar el scroll inferior */\n  shouldShowBottomScroll(): boolean {\n    const dir = this.mergedProps.direction;\n    return (dir === 'bottom' || dir === 'both') && this.items().length > 0;\n  }\n\n  /** Carga inicial de datos */\n  async loadInitial(): Promise<void> {\n    if (!this.props.dataSource.loadFn) return;\n\n    this.state.set('loading');\n    this.stateChange.emit('loading');\n    this.error.set(null);\n\n    try {\n      const params: LoadParams = {\n        direction: 'bottom',\n        page: 0,\n        pageSize: this.mergedProps.pageSize ?? 20,\n      };\n\n      const result = await this.executeLoad(params);\n\n      this.items.set(result.items as T[]);\n      this.hasMoreBottom.set(result.hasMore);\n      this.currentPage = 1;\n      this.currentCursor = result.cursor;\n      this.isInitialLoad.set(false);\n\n      this.state.set('idle');\n      this.stateChange.emit('idle');\n      this.itemsChange.emit(this.items());\n    } catch (err) {\n      this.handleError(err as Error);\n    }\n  }\n\n  /** Cargar mas items en la parte inferior */\n  async loadBottom(): Promise<void> {\n    if (!this.hasMoreBottom() || this.state() === 'loading') return;\n    if (!this.props.dataSource.loadFn) return;\n\n    this.state.set('loading');\n    this.stateChange.emit('loading');\n\n    try {\n      const params: LoadParams = {\n        direction: 'bottom',\n        page: this.currentPage,\n        pageSize: this.mergedProps.pageSize ?? 20,\n        cursor: this.currentCursor,\n        lastItem: this.items()[this.items().length - 1],\n      };\n\n      const result = await this.executeLoad(params);\n\n      this.items.update((current) => [...current, ...(result.items as T[])]);\n      this.hasMoreBottom.set(result.hasMore);\n      this.currentPage++;\n      this.currentCursor = result.cursor;\n\n      this.state.set(result.hasMore ? 'idle' : 'complete');\n      this.stateChange.emit(result.hasMore ? 'idle' : 'complete');\n      this.itemsChange.emit(this.items());\n    } catch (err) {\n      this.handleError(err as Error);\n    }\n  }\n\n  /** Cargar mas items en la parte superior */\n  async loadTop(): Promise<void> {\n    if (!this.hasMoreTop() || this.state() === 'loading') return;\n    if (!this.props.dataSource.loadFn) return;\n\n    this.state.set('loading');\n\n    try {\n      const params: LoadParams = {\n        direction: 'top',\n        page: 0,\n        pageSize: this.mergedProps.pageSize ?? 20,\n        firstItem: this.items()[0],\n      };\n\n      const result = await this.executeLoad(params);\n\n      this.items.update((current) => [...(result.items as T[]), ...current]);\n      this.hasMoreTop.set(result.hasMore);\n\n      this.state.set('idle');\n      this.itemsChange.emit(this.items());\n    } catch (err) {\n      this.handleError(err as Error);\n    }\n  }\n\n  /** Refresh - recargar desde cero */\n  async refreshList(): Promise<void> {\n    this.currentPage = 0;\n    this.currentCursor = null;\n    this.hasMoreBottom.set(true);\n    this.items.set([]);\n    await this.loadInitial();\n  }\n\n  /** Reintentar despues de error */\n  async retry(): Promise<void> {\n    this.error.set(null);\n    if (this.items().length === 0) {\n      await this.loadInitial();\n    } else {\n      await this.loadBottom();\n    }\n  }\n\n  /** Reset completo */\n  async reset(): Promise<void> {\n    this.items.set([]);\n    this.currentPage = 0;\n    this.currentCursor = null;\n    this.hasMoreBottom.set(true);\n    this.hasMoreTop.set(false);\n    this.error.set(null);\n    this.isInitialLoad.set(true);\n    this.state.set('idle');\n\n    if (this.mergedProps.autoLoad) {\n      await this.loadInitial();\n    }\n  }\n\n  /** Agregar items al inicio */\n  prependItems(newItems: T[]): void {\n    this.items.update((current) => [...newItems, ...current]);\n    this.itemsChange.emit(this.items());\n  }\n\n  /** Agregar items al final */\n  appendItems(newItems: T[]): void {\n    this.items.update((current) => [...current, ...newItems]);\n    this.itemsChange.emit(this.items());\n  }\n\n  /** Actualizar un item por indice */\n  updateItem(index: number, item: T): void {\n    this.items.update((current) => {\n      const updated = [...current];\n      updated[index] = item;\n      return updated;\n    });\n    this.itemsChange.emit(this.items());\n  }\n\n  /** Remover un item por indice */\n  removeItem(index: number): void {\n    this.items.update((current) => current.filter((_, i) => i !== index));\n    this.itemsChange.emit(this.items());\n  }\n\n  /** Handler para evento de infinite scroll */\n  async onInfiniteScroll(event: CustomEvent): Promise<void> {\n    await this.loadBottom();\n    (event.target as HTMLIonInfiniteScrollElement).complete();\n  }\n\n  /** Handler para evento de refresh */\n  async onRefreshTriggered(event: RefreshEvent): Promise<void> {\n    this.refresh.emit(event);\n    await this.refreshList();\n    event.complete();\n  }\n\n  private async executeLoad(params: LoadParams): Promise<LoadResult<unknown>> {\n    const loadFn = this.props.dataSource.loadFn!;\n    const result = loadFn(params);\n\n    if (isObservable(result)) {\n      return await firstValueFrom(result);\n    }\n    return await result;\n  }\n\n  private handleError(err: Error): void {\n    this.error.set(err);\n    this.state.set('error');\n    this.stateChange.emit('error');\n    this.errorOccurred.emit(err);\n    console.error('[InfiniteList] Error loading items:', err);\n  }\n}\n"]}
|