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.
Files changed (42) hide show
  1. package/esm2022/lib/components/molecules/refresher/refresher.component.mjs +254 -0
  2. package/esm2022/lib/components/molecules/refresher/types.mjs +15 -0
  3. package/esm2022/lib/components/organisms/infinite-list/infinite-list.component.mjs +618 -0
  4. package/esm2022/lib/components/organisms/infinite-list/types.mjs +15 -0
  5. package/esm2022/lib/components/templates/page-template/page-template.component.mjs +5 -5
  6. package/esm2022/lib/services/pagination/index.mjs +5 -0
  7. package/esm2022/lib/services/pagination/pagination.service.mjs +218 -0
  8. package/esm2022/lib/services/pagination/types.mjs +14 -0
  9. package/esm2022/lib/services/skeleton/config.mjs +79 -0
  10. package/esm2022/lib/services/skeleton/directives/loading.directive.mjs +215 -0
  11. package/esm2022/lib/services/skeleton/index.mjs +16 -0
  12. package/esm2022/lib/services/skeleton/skeleton.service.mjs +198 -0
  13. package/esm2022/lib/services/skeleton/templates/detail-skeleton.component.mjs +223 -0
  14. package/esm2022/lib/services/skeleton/templates/form-skeleton.component.mjs +127 -0
  15. package/esm2022/lib/services/skeleton/templates/grid-skeleton.component.mjs +154 -0
  16. package/esm2022/lib/services/skeleton/templates/list-skeleton.component.mjs +110 -0
  17. package/esm2022/lib/services/skeleton/templates/profile-skeleton.component.mjs +207 -0
  18. package/esm2022/lib/services/skeleton/templates/table-skeleton.component.mjs +116 -0
  19. package/esm2022/lib/services/skeleton/types.mjs +11 -0
  20. package/esm2022/public-api.mjs +12 -1
  21. package/fesm2022/valtech-components.mjs +3887 -1370
  22. package/fesm2022/valtech-components.mjs.map +1 -1
  23. package/lib/components/molecules/refresher/refresher.component.d.ts +79 -0
  24. package/lib/components/molecules/refresher/types.d.ts +86 -0
  25. package/lib/components/organisms/infinite-list/infinite-list.component.d.ts +111 -0
  26. package/lib/components/organisms/infinite-list/types.d.ts +197 -0
  27. package/lib/services/pagination/index.d.ts +2 -0
  28. package/lib/services/pagination/pagination.service.d.ts +43 -0
  29. package/lib/services/pagination/types.d.ts +113 -0
  30. package/lib/services/skeleton/config.d.ts +30 -0
  31. package/lib/services/skeleton/directives/loading.directive.d.ts +71 -0
  32. package/lib/services/skeleton/index.d.ts +10 -0
  33. package/lib/services/skeleton/skeleton.service.d.ts +127 -0
  34. package/lib/services/skeleton/templates/detail-skeleton.component.d.ts +18 -0
  35. package/lib/services/skeleton/templates/form-skeleton.component.d.ts +22 -0
  36. package/lib/services/skeleton/templates/grid-skeleton.component.d.ts +18 -0
  37. package/lib/services/skeleton/templates/list-skeleton.component.d.ts +17 -0
  38. package/lib/services/skeleton/templates/profile-skeleton.component.d.ts +20 -0
  39. package/lib/services/skeleton/templates/table-skeleton.component.d.ts +17 -0
  40. package/lib/services/skeleton/types.d.ts +111 -0
  41. package/package.json +1 -1
  42. 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"]}