valtech-components 2.0.529 → 2.0.530

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.
@@ -1,6 +1,7 @@
1
- import { Component, Input, Output, EventEmitter } from '@angular/core';
1
+ import { Component, Input, Output, EventEmitter, inject } from '@angular/core';
2
2
  import { CommonModule } from '@angular/common';
3
3
  import { FormsModule } from '@angular/forms';
4
+ import { I18nService } from '../../../services/i18n';
4
5
  import { IonIcon, IonButton, IonBadge, IonTextarea, IonItem, IonSelect, IonSelectOption, IonSpinner, IonInfiniteScroll, IonInfiniteScrollContent, } from '@ionic/angular/standalone';
5
6
  import { addIcons } from 'ionicons';
6
7
  import { chatbubblesOutline, filterOutline, sendOutline, chatbubbleEllipsesOutline, swapVerticalOutline, } from 'ionicons/icons';
@@ -61,6 +62,7 @@ addIcons({
61
62
  */
62
63
  export class CommentSectionComponent {
63
64
  constructor() {
65
+ this.i18n = inject(I18nService);
64
66
  // Section events
65
67
  this.sortChange = new EventEmitter();
66
68
  this.commentSubmit = new EventEmitter();
@@ -84,8 +86,11 @@ export class CommentSectionComponent {
84
86
  this.updateDisplayTexts();
85
87
  }
86
88
  updateDisplayTexts() {
87
- this.displayTitle = this.props.title || 'Comentarios';
88
- this.displayLoadMoreLabel = this.props.loadMoreLabel || 'Cargar más comentarios';
89
+ this.displayTitle = this.props.title || this.i18n.t('comments');
90
+ this.displayLoadMoreLabel = this.props.loadMoreLabel || this.i18n.t('loadMoreComments');
91
+ }
92
+ getSortByLabel() {
93
+ return this.i18n.t('sortBy');
89
94
  }
90
95
  formatCount(count) {
91
96
  if (count >= 1000000) {
@@ -100,16 +105,16 @@ export class CommentSectionComponent {
100
105
  return option.label;
101
106
  }
102
107
  getInputPlaceholder() {
103
- return this.props.inputConfig?.placeholder || 'Escribe un comentario...';
108
+ return this.props.inputConfig?.placeholder || this.i18n.t('writeComment');
104
109
  }
105
110
  getSubmitLabel() {
106
- return this.props.inputConfig?.submitLabel || 'Publicar';
111
+ return this.props.inputConfig?.submitLabel || this.i18n.t('publish');
107
112
  }
108
113
  getEmptyTitle() {
109
- return this.props.emptyState?.title || 'Sin comentarios aún';
114
+ return this.props.emptyState?.title || this.i18n.t('noCommentsYet');
110
115
  }
111
116
  getEmptyMessage() {
112
- return this.props.emptyState?.message || '¡Sé el primero en compartir tu opinión!';
117
+ return this.props.emptyState?.message || this.i18n.t('beFirstToComment');
113
118
  }
114
119
  getSkeletonArray() {
115
120
  const count = this.props.skeletonCount || 3;
@@ -205,7 +210,7 @@ export class CommentSectionComponent {
205
210
  <ion-icon name="swap-vertical-outline" slot="start" class="sort-icon"></ion-icon>
206
211
  <ion-select
207
212
  [value]="props.selectedSort"
208
- [placeholder]="props.sortLabel || 'Sort by'"
213
+ [placeholder]="props.sortLabel || getSortByLabel()"
209
214
  interface="popover"
210
215
  (ionChange)="onSortChange($event)"
211
216
  >
@@ -375,7 +380,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
375
380
  <ion-icon name="swap-vertical-outline" slot="start" class="sort-icon"></ion-icon>
376
381
  <ion-select
377
382
  [value]="props.selectedSort"
378
- [placeholder]="props.sortLabel || 'Sort by'"
383
+ [placeholder]="props.sortLabel || getSortByLabel()"
379
384
  interface="popover"
380
385
  (ionChange)="onSortChange($event)"
381
386
  >
@@ -529,4 +534,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
529
534
  }], replyStart: [{
530
535
  type: Output
531
536
  }] } });
532
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"comment-section.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/organisms/comment-section/comment-section.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, Output, EventEmitter, OnInit } from '@angular/core';\nimport { CommonModule } from '@angular/common';\nimport { FormsModule } from '@angular/forms';\nimport {\n  IonIcon,\n  IonButton,\n  IonBadge,\n  IonTextarea,\n  IonItem,\n  IonSelect,\n  IonSelectOption,\n  IonSpinner,\n  IonInfiniteScroll,\n  IonInfiniteScrollContent,\n} from '@ionic/angular/standalone';\nimport { addIcons } from 'ionicons';\nimport {\n  chatbubblesOutline,\n  filterOutline,\n  sendOutline,\n  chatbubbleEllipsesOutline,\n  swapVerticalOutline,\n} from 'ionicons/icons';\n\nimport { CommentComponent } from '../../molecules/comment/comment.component';\nimport { AvatarComponent } from '../../atoms/avatar/avatar.component';\nimport { SkeletonComponent } from '../../atoms/skeleton/skeleton.component';\nimport {\n  CommentSectionMetadata,\n  CommentSortOption,\n  CommentSortChangeEvent,\n  CommentSubmitEvent,\n  CommentSectionLoadMoreEvent,\n} from './types';\nimport {\n  CommentAuthorClickEvent,\n  CommentReactionClickEvent,\n  CommentActionClickEvent,\n  CommentMenuItemClickEvent,\n  CommentLoadMoreEvent,\n} from '../../molecules/comment/types';\n\naddIcons({\n  chatbubblesOutline,\n  filterOutline,\n  sendOutline,\n  chatbubbleEllipsesOutline,\n  swapVerticalOutline,\n});\n\n@Component({\n  selector: 'val-comment-section',\n  standalone: true,\n  imports: [\n    CommonModule,\n    FormsModule,\n    IonIcon,\n    IonButton,\n    IonBadge,\n    IonTextarea,\n    IonItem,\n    IonSelect,\n    IonSelectOption,\n    IonSpinner,\n    IonInfiniteScroll,\n    IonInfiniteScrollContent,\n    CommentComponent,\n    AvatarComponent,\n    SkeletonComponent,\n  ],\n  template: `\n    <div class=\"comment-section\" [class.loading]=\"props.loading\">\n      <!-- Header -->\n      <div class=\"section-header\">\n        <div class=\"header-title\">\n          <ion-icon name=\"chatbubbles-outline\" class=\"title-icon\"></ion-icon>\n          <h3 class=\"title\">{{ displayTitle }}</h3>\n          @if (props.showCount !== false && props.count !== undefined) {\n            <ion-badge color=\"medium\" class=\"count-badge\">\n              {{ formatCount(props.count) }}\n            </ion-badge>\n          }\n        </div>\n\n        @if (props.sortOptions && props.sortOptions.length > 0) {\n          <div class=\"header-actions\">\n            <ion-item lines=\"none\" class=\"sort-select-item\">\n              <ion-icon name=\"swap-vertical-outline\" slot=\"start\" class=\"sort-icon\"></ion-icon>\n              <ion-select\n                [value]=\"props.selectedSort\"\n                [placeholder]=\"props.sortLabel || 'Sort by'\"\n                interface=\"popover\"\n                (ionChange)=\"onSortChange($event)\"\n              >\n                @for (option of props.sortOptions; track option.token) {\n                  <ion-select-option [value]=\"option.token\">\n                    {{ getSortOptionLabel(option) }}\n                  </ion-select-option>\n                }\n              </ion-select>\n            </ion-item>\n          </div>\n        }\n      </div>\n\n      <!-- New Comment Input -->\n      @if (props.showInput !== false) {\n        <div class=\"new-comment-section\">\n          <div class=\"input-wrapper\">\n            @if (props.inputConfig?.currentUser?.avatar) {\n              <div class=\"input-avatar\">\n                <val-avatar [props]=\"props.inputConfig.currentUser.avatar\"></val-avatar>\n              </div>\n            }\n\n            <div class=\"input-container\">\n              <ion-textarea\n                [(ngModel)]=\"newCommentText\"\n                [placeholder]=\"getInputPlaceholder()\"\n                [maxlength]=\"props.inputConfig?.maxLength || 2000\"\n                [disabled]=\"props.inputConfig?.disabled || props.loading\"\n                [autoGrow]=\"true\"\n                rows=\"2\"\n                class=\"comment-textarea\"\n              ></ion-textarea>\n\n              <div class=\"input-actions\">\n                @if (props.inputConfig?.showCounter && props.inputConfig?.maxLength) {\n                  <span class=\"char-counter\" [class.warning]=\"isNearLimit()\">\n                    {{ newCommentText.length }} / {{ props.inputConfig.maxLength }}\n                  </span>\n                }\n\n                <ion-button\n                  [color]=\"props.inputConfig?.submitColor || props.color || 'primary'\"\n                  [disabled]=\"!canSubmit()\"\n                  size=\"small\"\n                  (click)=\"onSubmitComment()\"\n                >\n                  <ion-icon name=\"send-outline\" slot=\"start\"></ion-icon>\n                  {{ getSubmitLabel() }}\n                </ion-button>\n              </div>\n            </div>\n          </div>\n        </div>\n      }\n\n      @if (props.loading) {\n        <div class=\"loading-state\">\n          @for (i of getSkeletonArray(); track i) {\n            <div class=\"skeleton-comment\">\n              <val-skeleton [props]=\"{ type: 'avatar', width: '36px', height: '36px' }\"></val-skeleton>\n              <div class=\"skeleton-content\">\n                <val-skeleton [props]=\"{ type: 'text', width: '120px', height: '14px' }\"></val-skeleton>\n                <val-skeleton [props]=\"{ type: 'paragraph', lines: 2 }\"></val-skeleton>\n              </div>\n            </div>\n          }\n        </div>\n      } @else if (props.comments && props.comments.length > 0) {\n        <div class=\"comments-list\" [class.with-dividers]=\"props.showDividers\">\n          @for (comment of props.comments; track comment.token) {\n            <val-comment\n              [props]=\"comment\"\n              (authorClick)=\"authorClick.emit($event)\"\n              (reactionClick)=\"reactionClick.emit($event)\"\n              (actionClick)=\"onCommentAction($event)\"\n              (menuItemClick)=\"menuItemClick.emit($event)\"\n              (loadMoreClick)=\"commentLoadMore.emit($event)\"\n              (collapseToggle)=\"collapseToggle.emit($event)\"\n            ></val-comment>\n\n            @if (props.showDividers && !$last) {\n              <div class=\"comment-divider\"></div>\n            }\n          }\n\n          @if (props.hasMore && props.paginationMode !== 'infinite') {\n            <div class=\"load-more-section\">\n              @if (props.loadingMore) {\n                <ion-spinner name=\"crescent\" [color]=\"props.color || 'primary'\"></ion-spinner>\n              } @else {\n                <ion-button\n                  fill=\"outline\"\n                  [color]=\"props.color || 'primary'\"\n                  expand=\"block\"\n                  (click)=\"onLoadMore()\"\n                >\n                  {{ displayLoadMoreLabel }}\n                </ion-button>\n              }\n            </div>\n          }\n        </div>\n\n        @if (props.paginationMode === 'infinite') {\n          <ion-infinite-scroll\n            [threshold]=\"props.infiniteScrollThreshold || '100px'\"\n            [position]=\"props.infiniteScrollPosition || 'bottom'\"\n            [disabled]=\"!props.hasMore\"\n            (ionInfinite)=\"onInfiniteScroll($event)\"\n          >\n            <ion-infinite-scroll-content\n              [loadingSpinner]=\"'crescent'\"\n              [loadingText]=\"displayLoadMoreLabel\"\n            ></ion-infinite-scroll-content>\n          </ion-infinite-scroll>\n        }\n      } @else {\n        <div class=\"empty-state\">\n          <ion-icon\n            [name]=\"props.emptyState?.icon || 'chatbubble-ellipses-outline'\"\n            class=\"empty-icon\"\n          ></ion-icon>\n          <h4 class=\"empty-title\">{{ getEmptyTitle() }}</h4>\n          <p class=\"empty-message\">{{ getEmptyMessage() }}</p>\n        </div>\n      }\n    </div>\n  `,\n  styleUrls: ['./comment-section.component.scss'],\n})\n/**\n * val-comment-section\n *\n * An organism component that provides a complete comment section with:\n * - Header with title and count\n * - Sort/filter options\n * - New comment input\n * - Comments list with val-comment\n * - Load more pagination\n * - Empty state\n *\n * @example Basic usage\n * <val-comment-section [props]=\"{\n *   title: 'Comments',\n *   count: 42,\n *   comments: commentsArray,\n *   showInput: true\n * }\"></val-comment-section>\n *\n * @example With sorting\n * <val-comment-section [props]=\"{\n *   title: 'Reviews',\n *   count: 128,\n *   comments: reviews,\n *   sortOptions: [\n *     { token: 'newest', label: 'Newest first' },\n *     { token: 'oldest', label: 'Oldest first' },\n *     { token: 'popular', label: 'Most popular' }\n *   ],\n *   selectedSort: 'newest'\n * }\" (sortChange)=\"onSort($event)\"></val-comment-section>\n *\n * @input props: CommentSectionMetadata - Configuration for the section\n * @output sortChange - Sort option changed\n * @output commentSubmit - New comment submitted\n * @output loadMore - Load more clicked\n * @output authorClick - Comment author clicked (bubbled from val-comment)\n * @output reactionClick - Reaction clicked (bubbled)\n * @output actionClick - Action clicked (bubbled)\n * @output menuItemClick - Menu item clicked (bubbled)\n * @output commentLoadMore - Load more replies clicked (bubbled)\n * @output collapseToggle - Comment collapse toggled (bubbled)\n */\nexport class CommentSectionComponent implements OnInit {\n  @Input() props: CommentSectionMetadata;\n\n  // Section events\n  @Output() sortChange = new EventEmitter<CommentSortChangeEvent>();\n  @Output() commentSubmit = new EventEmitter<CommentSubmitEvent>();\n  @Output() loadMore = new EventEmitter<CommentSectionLoadMoreEvent>();\n\n  // Bubbled events from val-comment\n  @Output() authorClick = new EventEmitter<CommentAuthorClickEvent>();\n  @Output() reactionClick = new EventEmitter<CommentReactionClickEvent>();\n  @Output() actionClick = new EventEmitter<CommentActionClickEvent>();\n  @Output() menuItemClick = new EventEmitter<CommentMenuItemClickEvent>();\n  @Output() commentLoadMore = new EventEmitter<CommentLoadMoreEvent>();\n  @Output() collapseToggle = new EventEmitter<{ token: string; collapsed: boolean }>();\n\n  // Reply state\n  @Output() replyStart = new EventEmitter<{ commentToken: string }>();\n\n  newCommentText = '';\n  replyingTo: string | null = null;\n\n  displayTitle = '';\n  displayLoadMoreLabel = '';\n\n  ngOnInit(): void {\n    this.updateDisplayTexts();\n  }\n\n  private updateDisplayTexts(): void {\n    this.displayTitle = this.props.title || 'Comentarios';\n    this.displayLoadMoreLabel = this.props.loadMoreLabel || 'Cargar más comentarios';\n  }\n\n  formatCount(count: number): string {\n    if (count >= 1000000) {\n      return (count / 1000000).toFixed(1).replace(/\\.0$/, '') + 'M';\n    }\n    if (count >= 1000) {\n      return (count / 1000).toFixed(1).replace(/\\.0$/, '') + 'K';\n    }\n    return count.toString();\n  }\n\n  getSortOptionLabel(option: CommentSortOption): string {\n    return option.label;\n  }\n\n  getInputPlaceholder(): string {\n    return this.props.inputConfig?.placeholder || 'Escribe un comentario...';\n  }\n\n  getSubmitLabel(): string {\n    return this.props.inputConfig?.submitLabel || 'Publicar';\n  }\n\n  getEmptyTitle(): string {\n    return this.props.emptyState?.title || 'Sin comentarios aún';\n  }\n\n  getEmptyMessage(): string {\n    return this.props.emptyState?.message || '¡Sé el primero en compartir tu opinión!';\n  }\n\n  getSkeletonArray(): number[] {\n    const count = this.props.skeletonCount || 3;\n    return Array(count).fill(0).map((_, i) => i);\n  }\n\n  canSubmit(): boolean {\n    const minLength = this.props.inputConfig?.minLength || 1;\n    return (\n      this.newCommentText.trim().length >= minLength &&\n      !this.props.loading &&\n      !this.props.inputConfig?.disabled\n    );\n  }\n\n  isNearLimit(): boolean {\n    const maxLength = this.props.inputConfig?.maxLength || 2000;\n    return this.newCommentText.length > maxLength * 0.9;\n  }\n\n  onSortChange(event: CustomEvent): void {\n    const selectedToken = event.detail.value;\n    const option = this.props.sortOptions?.find((o) => o.token === selectedToken);\n\n    if (option) {\n      this.sortChange.emit({\n        option,\n        previousSort: this.props.selectedSort,\n      });\n    }\n  }\n\n  onSubmitComment(): void {\n    if (!this.canSubmit()) return;\n\n    this.commentSubmit.emit({\n      content: this.newCommentText.trim(),\n      parentToken: this.replyingTo || undefined,\n      sectionToken: this.props.token,\n    });\n\n    this.newCommentText = '';\n    this.replyingTo = null;\n  }\n\n  onCommentAction(event: CommentActionClickEvent): void {\n    // Check if it's a reply action\n    if (event.action.token === 'reply') {\n      this.replyingTo = event.commentToken;\n      this.replyStart.emit({ commentToken: event.commentToken });\n    }\n\n    this.actionClick.emit(event);\n  }\n\n  onLoadMore(): void {\n    this.loadMore.emit({\n      sectionToken: this.props.token,\n      currentCount: this.props.comments?.length || 0,\n    });\n  }\n\n  onInfiniteScroll(event: CustomEvent): void {\n    this.loadMore.emit({\n      sectionToken: this.props.token,\n      currentCount: this.props.comments?.length || 0,\n    });\n\n    // The parent component should call completeInfiniteScroll() when done loading\n    // Store reference to complete the infinite scroll\n    this.infiniteScrollEvent = event;\n  }\n\n  // Call this method from parent after loading more comments\n  completeInfiniteScroll(): void {\n    if (this.infiniteScrollEvent) {\n      (this.infiniteScrollEvent.target as HTMLIonInfiniteScrollElement).complete();\n      this.infiniteScrollEvent = null;\n    }\n  }\n\n  private infiniteScrollEvent: CustomEvent | null = null;\n\n  // Method to programmatically start a reply\n  startReply(commentToken: string): void {\n    this.replyingTo = commentToken;\n  }\n\n  // Method to cancel reply\n  cancelReply(): void {\n    this.replyingTo = null;\n  }\n}\n"]}
537
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"comment-section.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/organisms/comment-section/comment-section.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, Output, EventEmitter, OnInit, inject } from '@angular/core';\nimport { CommonModule } from '@angular/common';\nimport { FormsModule } from '@angular/forms';\nimport { I18nService } from '../../../services/i18n';\nimport {\n  IonIcon,\n  IonButton,\n  IonBadge,\n  IonTextarea,\n  IonItem,\n  IonSelect,\n  IonSelectOption,\n  IonSpinner,\n  IonInfiniteScroll,\n  IonInfiniteScrollContent,\n} from '@ionic/angular/standalone';\nimport { addIcons } from 'ionicons';\nimport {\n  chatbubblesOutline,\n  filterOutline,\n  sendOutline,\n  chatbubbleEllipsesOutline,\n  swapVerticalOutline,\n} from 'ionicons/icons';\n\nimport { CommentComponent } from '../../molecules/comment/comment.component';\nimport { AvatarComponent } from '../../atoms/avatar/avatar.component';\nimport { SkeletonComponent } from '../../atoms/skeleton/skeleton.component';\nimport {\n  CommentSectionMetadata,\n  CommentSortOption,\n  CommentSortChangeEvent,\n  CommentSubmitEvent,\n  CommentSectionLoadMoreEvent,\n} from './types';\nimport {\n  CommentAuthorClickEvent,\n  CommentReactionClickEvent,\n  CommentActionClickEvent,\n  CommentMenuItemClickEvent,\n  CommentLoadMoreEvent,\n} from '../../molecules/comment/types';\n\naddIcons({\n  chatbubblesOutline,\n  filterOutline,\n  sendOutline,\n  chatbubbleEllipsesOutline,\n  swapVerticalOutline,\n});\n\n@Component({\n  selector: 'val-comment-section',\n  standalone: true,\n  imports: [\n    CommonModule,\n    FormsModule,\n    IonIcon,\n    IonButton,\n    IonBadge,\n    IonTextarea,\n    IonItem,\n    IonSelect,\n    IonSelectOption,\n    IonSpinner,\n    IonInfiniteScroll,\n    IonInfiniteScrollContent,\n    CommentComponent,\n    AvatarComponent,\n    SkeletonComponent,\n  ],\n  template: `\n    <div class=\"comment-section\" [class.loading]=\"props.loading\">\n      <!-- Header -->\n      <div class=\"section-header\">\n        <div class=\"header-title\">\n          <ion-icon name=\"chatbubbles-outline\" class=\"title-icon\"></ion-icon>\n          <h3 class=\"title\">{{ displayTitle }}</h3>\n          @if (props.showCount !== false && props.count !== undefined) {\n            <ion-badge color=\"medium\" class=\"count-badge\">\n              {{ formatCount(props.count) }}\n            </ion-badge>\n          }\n        </div>\n\n        @if (props.sortOptions && props.sortOptions.length > 0) {\n          <div class=\"header-actions\">\n            <ion-item lines=\"none\" class=\"sort-select-item\">\n              <ion-icon name=\"swap-vertical-outline\" slot=\"start\" class=\"sort-icon\"></ion-icon>\n              <ion-select\n                [value]=\"props.selectedSort\"\n                [placeholder]=\"props.sortLabel || getSortByLabel()\"\n                interface=\"popover\"\n                (ionChange)=\"onSortChange($event)\"\n              >\n                @for (option of props.sortOptions; track option.token) {\n                  <ion-select-option [value]=\"option.token\">\n                    {{ getSortOptionLabel(option) }}\n                  </ion-select-option>\n                }\n              </ion-select>\n            </ion-item>\n          </div>\n        }\n      </div>\n\n      <!-- New Comment Input -->\n      @if (props.showInput !== false) {\n        <div class=\"new-comment-section\">\n          <div class=\"input-wrapper\">\n            @if (props.inputConfig?.currentUser?.avatar) {\n              <div class=\"input-avatar\">\n                <val-avatar [props]=\"props.inputConfig.currentUser.avatar\"></val-avatar>\n              </div>\n            }\n\n            <div class=\"input-container\">\n              <ion-textarea\n                [(ngModel)]=\"newCommentText\"\n                [placeholder]=\"getInputPlaceholder()\"\n                [maxlength]=\"props.inputConfig?.maxLength || 2000\"\n                [disabled]=\"props.inputConfig?.disabled || props.loading\"\n                [autoGrow]=\"true\"\n                rows=\"2\"\n                class=\"comment-textarea\"\n              ></ion-textarea>\n\n              <div class=\"input-actions\">\n                @if (props.inputConfig?.showCounter && props.inputConfig?.maxLength) {\n                  <span class=\"char-counter\" [class.warning]=\"isNearLimit()\">\n                    {{ newCommentText.length }} / {{ props.inputConfig.maxLength }}\n                  </span>\n                }\n\n                <ion-button\n                  [color]=\"props.inputConfig?.submitColor || props.color || 'primary'\"\n                  [disabled]=\"!canSubmit()\"\n                  size=\"small\"\n                  (click)=\"onSubmitComment()\"\n                >\n                  <ion-icon name=\"send-outline\" slot=\"start\"></ion-icon>\n                  {{ getSubmitLabel() }}\n                </ion-button>\n              </div>\n            </div>\n          </div>\n        </div>\n      }\n\n      @if (props.loading) {\n        <div class=\"loading-state\">\n          @for (i of getSkeletonArray(); track i) {\n            <div class=\"skeleton-comment\">\n              <val-skeleton [props]=\"{ type: 'avatar', width: '36px', height: '36px' }\"></val-skeleton>\n              <div class=\"skeleton-content\">\n                <val-skeleton [props]=\"{ type: 'text', width: '120px', height: '14px' }\"></val-skeleton>\n                <val-skeleton [props]=\"{ type: 'paragraph', lines: 2 }\"></val-skeleton>\n              </div>\n            </div>\n          }\n        </div>\n      } @else if (props.comments && props.comments.length > 0) {\n        <div class=\"comments-list\" [class.with-dividers]=\"props.showDividers\">\n          @for (comment of props.comments; track comment.token) {\n            <val-comment\n              [props]=\"comment\"\n              (authorClick)=\"authorClick.emit($event)\"\n              (reactionClick)=\"reactionClick.emit($event)\"\n              (actionClick)=\"onCommentAction($event)\"\n              (menuItemClick)=\"menuItemClick.emit($event)\"\n              (loadMoreClick)=\"commentLoadMore.emit($event)\"\n              (collapseToggle)=\"collapseToggle.emit($event)\"\n            ></val-comment>\n\n            @if (props.showDividers && !$last) {\n              <div class=\"comment-divider\"></div>\n            }\n          }\n\n          @if (props.hasMore && props.paginationMode !== 'infinite') {\n            <div class=\"load-more-section\">\n              @if (props.loadingMore) {\n                <ion-spinner name=\"crescent\" [color]=\"props.color || 'primary'\"></ion-spinner>\n              } @else {\n                <ion-button\n                  fill=\"outline\"\n                  [color]=\"props.color || 'primary'\"\n                  expand=\"block\"\n                  (click)=\"onLoadMore()\"\n                >\n                  {{ displayLoadMoreLabel }}\n                </ion-button>\n              }\n            </div>\n          }\n        </div>\n\n        @if (props.paginationMode === 'infinite') {\n          <ion-infinite-scroll\n            [threshold]=\"props.infiniteScrollThreshold || '100px'\"\n            [position]=\"props.infiniteScrollPosition || 'bottom'\"\n            [disabled]=\"!props.hasMore\"\n            (ionInfinite)=\"onInfiniteScroll($event)\"\n          >\n            <ion-infinite-scroll-content\n              [loadingSpinner]=\"'crescent'\"\n              [loadingText]=\"displayLoadMoreLabel\"\n            ></ion-infinite-scroll-content>\n          </ion-infinite-scroll>\n        }\n      } @else {\n        <div class=\"empty-state\">\n          <ion-icon\n            [name]=\"props.emptyState?.icon || 'chatbubble-ellipses-outline'\"\n            class=\"empty-icon\"\n          ></ion-icon>\n          <h4 class=\"empty-title\">{{ getEmptyTitle() }}</h4>\n          <p class=\"empty-message\">{{ getEmptyMessage() }}</p>\n        </div>\n      }\n    </div>\n  `,\n  styleUrls: ['./comment-section.component.scss'],\n})\n/**\n * val-comment-section\n *\n * An organism component that provides a complete comment section with:\n * - Header with title and count\n * - Sort/filter options\n * - New comment input\n * - Comments list with val-comment\n * - Load more pagination\n * - Empty state\n *\n * @example Basic usage\n * <val-comment-section [props]=\"{\n *   title: 'Comments',\n *   count: 42,\n *   comments: commentsArray,\n *   showInput: true\n * }\"></val-comment-section>\n *\n * @example With sorting\n * <val-comment-section [props]=\"{\n *   title: 'Reviews',\n *   count: 128,\n *   comments: reviews,\n *   sortOptions: [\n *     { token: 'newest', label: 'Newest first' },\n *     { token: 'oldest', label: 'Oldest first' },\n *     { token: 'popular', label: 'Most popular' }\n *   ],\n *   selectedSort: 'newest'\n * }\" (sortChange)=\"onSort($event)\"></val-comment-section>\n *\n * @input props: CommentSectionMetadata - Configuration for the section\n * @output sortChange - Sort option changed\n * @output commentSubmit - New comment submitted\n * @output loadMore - Load more clicked\n * @output authorClick - Comment author clicked (bubbled from val-comment)\n * @output reactionClick - Reaction clicked (bubbled)\n * @output actionClick - Action clicked (bubbled)\n * @output menuItemClick - Menu item clicked (bubbled)\n * @output commentLoadMore - Load more replies clicked (bubbled)\n * @output collapseToggle - Comment collapse toggled (bubbled)\n */\nexport class CommentSectionComponent implements OnInit {\n  private i18n = inject(I18nService);\n\n  @Input() props: CommentSectionMetadata;\n\n  // Section events\n  @Output() sortChange = new EventEmitter<CommentSortChangeEvent>();\n  @Output() commentSubmit = new EventEmitter<CommentSubmitEvent>();\n  @Output() loadMore = new EventEmitter<CommentSectionLoadMoreEvent>();\n\n  // Bubbled events from val-comment\n  @Output() authorClick = new EventEmitter<CommentAuthorClickEvent>();\n  @Output() reactionClick = new EventEmitter<CommentReactionClickEvent>();\n  @Output() actionClick = new EventEmitter<CommentActionClickEvent>();\n  @Output() menuItemClick = new EventEmitter<CommentMenuItemClickEvent>();\n  @Output() commentLoadMore = new EventEmitter<CommentLoadMoreEvent>();\n  @Output() collapseToggle = new EventEmitter<{ token: string; collapsed: boolean }>();\n\n  // Reply state\n  @Output() replyStart = new EventEmitter<{ commentToken: string }>();\n\n  newCommentText = '';\n  replyingTo: string | null = null;\n\n  displayTitle = '';\n  displayLoadMoreLabel = '';\n\n  ngOnInit(): void {\n    this.updateDisplayTexts();\n  }\n\n  private updateDisplayTexts(): void {\n    this.displayTitle = this.props.title || this.i18n.t('comments');\n    this.displayLoadMoreLabel = this.props.loadMoreLabel || this.i18n.t('loadMoreComments');\n  }\n\n  getSortByLabel(): string {\n    return this.i18n.t('sortBy');\n  }\n\n  formatCount(count: number): string {\n    if (count >= 1000000) {\n      return (count / 1000000).toFixed(1).replace(/\\.0$/, '') + 'M';\n    }\n    if (count >= 1000) {\n      return (count / 1000).toFixed(1).replace(/\\.0$/, '') + 'K';\n    }\n    return count.toString();\n  }\n\n  getSortOptionLabel(option: CommentSortOption): string {\n    return option.label;\n  }\n\n  getInputPlaceholder(): string {\n    return this.props.inputConfig?.placeholder || this.i18n.t('writeComment');\n  }\n\n  getSubmitLabel(): string {\n    return this.props.inputConfig?.submitLabel || this.i18n.t('publish');\n  }\n\n  getEmptyTitle(): string {\n    return this.props.emptyState?.title || this.i18n.t('noCommentsYet');\n  }\n\n  getEmptyMessage(): string {\n    return this.props.emptyState?.message || this.i18n.t('beFirstToComment');\n  }\n\n  getSkeletonArray(): number[] {\n    const count = this.props.skeletonCount || 3;\n    return Array(count).fill(0).map((_, i) => i);\n  }\n\n  canSubmit(): boolean {\n    const minLength = this.props.inputConfig?.minLength || 1;\n    return (\n      this.newCommentText.trim().length >= minLength &&\n      !this.props.loading &&\n      !this.props.inputConfig?.disabled\n    );\n  }\n\n  isNearLimit(): boolean {\n    const maxLength = this.props.inputConfig?.maxLength || 2000;\n    return this.newCommentText.length > maxLength * 0.9;\n  }\n\n  onSortChange(event: CustomEvent): void {\n    const selectedToken = event.detail.value;\n    const option = this.props.sortOptions?.find((o) => o.token === selectedToken);\n\n    if (option) {\n      this.sortChange.emit({\n        option,\n        previousSort: this.props.selectedSort,\n      });\n    }\n  }\n\n  onSubmitComment(): void {\n    if (!this.canSubmit()) return;\n\n    this.commentSubmit.emit({\n      content: this.newCommentText.trim(),\n      parentToken: this.replyingTo || undefined,\n      sectionToken: this.props.token,\n    });\n\n    this.newCommentText = '';\n    this.replyingTo = null;\n  }\n\n  onCommentAction(event: CommentActionClickEvent): void {\n    // Check if it's a reply action\n    if (event.action.token === 'reply') {\n      this.replyingTo = event.commentToken;\n      this.replyStart.emit({ commentToken: event.commentToken });\n    }\n\n    this.actionClick.emit(event);\n  }\n\n  onLoadMore(): void {\n    this.loadMore.emit({\n      sectionToken: this.props.token,\n      currentCount: this.props.comments?.length || 0,\n    });\n  }\n\n  onInfiniteScroll(event: CustomEvent): void {\n    this.loadMore.emit({\n      sectionToken: this.props.token,\n      currentCount: this.props.comments?.length || 0,\n    });\n\n    // The parent component should call completeInfiniteScroll() when done loading\n    // Store reference to complete the infinite scroll\n    this.infiniteScrollEvent = event;\n  }\n\n  // Call this method from parent after loading more comments\n  completeInfiniteScroll(): void {\n    if (this.infiniteScrollEvent) {\n      (this.infiniteScrollEvent.target as HTMLIonInfiniteScrollElement).complete();\n      this.infiniteScrollEvent = null;\n    }\n  }\n\n  private infiniteScrollEvent: CustomEvent | null = null;\n\n  // Method to programmatically start a reply\n  startReply(commentToken: string): void {\n    this.replyingTo = commentToken;\n  }\n\n  // Method to cancel reply\n  cancelReply(): void {\n    this.replyingTo = null;\n  }\n}\n"]}
@@ -1,5 +1,6 @@
1
1
  import { Component, Input, Output, EventEmitter, signal, computed, ViewChild, inject, ChangeDetectorRef, } from '@angular/core';
2
2
  import { CommonModule } from '@angular/common';
3
+ import { I18nService } from '../../../services/i18n';
3
4
  import { IonInfiniteScroll, IonInfiniteScrollContent, IonButton, IonSpinner, IonIcon, IonText, IonList, IonItem, } from '@ionic/angular/standalone';
4
5
  import { firstValueFrom, isObservable } from 'rxjs';
5
6
  import { DEFAULT_INFINITE_LIST_METADATA, } from './types';
@@ -49,6 +50,7 @@ export class InfiniteListComponent {
49
50
  constructor() {
50
51
  this.skeletonService = inject(SkeletonService);
51
52
  this.cdr = inject(ChangeDetectorRef);
53
+ this.i18n = inject(I18nService);
52
54
  // === Events ===
53
55
  this.loadMore = new EventEmitter();
54
56
  this.refresh = new EventEmitter();
@@ -74,11 +76,11 @@ export class InfiniteListComponent {
74
76
  this.statusAnnouncement = computed(() => {
75
77
  switch (this.state()) {
76
78
  case 'loading':
77
- return 'Cargando items...';
79
+ return this.i18n.t('loadingItems');
78
80
  case 'error':
79
- return `Error: ${this.error()?.message || 'Ocurrio un error'}`;
81
+ return `${this.i18n.t('error')}: ${this.error()?.message || this.i18n.t('errorOccurred')}`;
80
82
  case 'complete':
81
- return 'Todos los items han sido cargados';
83
+ return this.i18n.t('allItemsLoaded');
82
84
  default:
83
85
  return '';
84
86
  }
@@ -362,11 +364,11 @@ export class InfiniteListComponent {
362
364
  } @else {
363
365
  <ion-icon name="alert-circle-outline" color="danger" size="large"></ion-icon>
364
366
  }
365
- <h3>{{ mergedProps.errorState?.title || 'Error' }}</h3>
366
- <p>{{ mergedProps.errorState?.message || error()?.message || 'Ocurrio un error' }}</p>
367
+ <h3>{{ mergedProps.errorState?.title || i18n.t('error') }}</h3>
368
+ <p>{{ mergedProps.errorState?.message || error()?.message || i18n.t('errorOccurred') }}</p>
367
369
  @if (mergedProps.errorState?.showRetry !== false) {
368
370
  <ion-button fill="outline" (click)="retry()">
369
- {{ mergedProps.errorState?.retryText || 'Reintentar' }}
371
+ {{ mergedProps.errorState?.retryText || i18n.t('retry') }}
370
372
  </ion-button>
371
373
  }
372
374
  }
@@ -407,10 +409,10 @@ export class InfiniteListComponent {
407
409
  @if (state() === 'loading') {
408
410
  <ion-spinner [name]="mergedProps.spinnerType" slot="start"></ion-spinner>
409
411
  }
410
- {{ mergedProps.loadMoreText || 'Cargar mas' }}
412
+ {{ mergedProps.loadMoreText || i18n.t('loadMore') }}
411
413
  </ion-button>
412
414
  } @else {
413
- <ion-text color="medium">{{ mergedProps.noMoreText || 'No hay mas items' }}</ion-text>
415
+ <ion-text color="medium">{{ mergedProps.noMoreText || i18n.t('noMoreItems') }}</ion-text>
414
416
  }
415
417
  </div>
416
418
  } @else {
@@ -421,7 +423,7 @@ export class InfiniteListComponent {
421
423
  >
422
424
  <ion-infinite-scroll-content
423
425
  [loadingSpinner]="mergedProps.spinnerType"
424
- [loadingText]="state() === 'loading' ? 'Cargando...' : ''"
426
+ [loadingText]="state() === 'loading' ? i18n.t('loading') : ''"
425
427
  ></ion-infinite-scroll-content>
426
428
  </ion-infinite-scroll>
427
429
  }
@@ -430,7 +432,7 @@ export class InfiniteListComponent {
430
432
  <!-- No more items indicator -->
431
433
  @if (!hasMoreBottom() && items().length > 0 && !mergedProps.useLoadMoreButton) {
432
434
  <div class="infinite-list-end">
433
- <ion-text color="medium">{{ mergedProps.noMoreText || 'No hay mas items' }}</ion-text>
435
+ <ion-text color="medium">{{ mergedProps.noMoreText || i18n.t('noMoreItems') }}</ion-text>
434
436
  </div>
435
437
  }
436
438
  </div>
@@ -520,11 +522,11 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
520
522
  } @else {
521
523
  <ion-icon name="alert-circle-outline" color="danger" size="large"></ion-icon>
522
524
  }
523
- <h3>{{ mergedProps.errorState?.title || 'Error' }}</h3>
524
- <p>{{ mergedProps.errorState?.message || error()?.message || 'Ocurrio un error' }}</p>
525
+ <h3>{{ mergedProps.errorState?.title || i18n.t('error') }}</h3>
526
+ <p>{{ mergedProps.errorState?.message || error()?.message || i18n.t('errorOccurred') }}</p>
525
527
  @if (mergedProps.errorState?.showRetry !== false) {
526
528
  <ion-button fill="outline" (click)="retry()">
527
- {{ mergedProps.errorState?.retryText || 'Reintentar' }}
529
+ {{ mergedProps.errorState?.retryText || i18n.t('retry') }}
528
530
  </ion-button>
529
531
  }
530
532
  }
@@ -565,10 +567,10 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
565
567
  @if (state() === 'loading') {
566
568
  <ion-spinner [name]="mergedProps.spinnerType" slot="start"></ion-spinner>
567
569
  }
568
- {{ mergedProps.loadMoreText || 'Cargar mas' }}
570
+ {{ mergedProps.loadMoreText || i18n.t('loadMore') }}
569
571
  </ion-button>
570
572
  } @else {
571
- <ion-text color="medium">{{ mergedProps.noMoreText || 'No hay mas items' }}</ion-text>
573
+ <ion-text color="medium">{{ mergedProps.noMoreText || i18n.t('noMoreItems') }}</ion-text>
572
574
  }
573
575
  </div>
574
576
  } @else {
@@ -579,7 +581,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
579
581
  >
580
582
  <ion-infinite-scroll-content
581
583
  [loadingSpinner]="mergedProps.spinnerType"
582
- [loadingText]="state() === 'loading' ? 'Cargando...' : ''"
584
+ [loadingText]="state() === 'loading' ? i18n.t('loading') : ''"
583
585
  ></ion-infinite-scroll-content>
584
586
  </ion-infinite-scroll>
585
587
  }
@@ -588,7 +590,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
588
590
  <!-- No more items indicator -->
589
591
  @if (!hasMoreBottom() && items().length > 0 && !mergedProps.useLoadMoreButton) {
590
592
  <div class="infinite-list-end">
591
- <ion-text color="medium">{{ mergedProps.noMoreText || 'No hay mas items' }}</ion-text>
593
+ <ion-text color="medium">{{ mergedProps.noMoreText || i18n.t('noMoreItems') }}</ion-text>
592
594
  </div>
593
595
  }
594
596
  </div>
@@ -615,4 +617,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
615
617
  }], errorOccurred: [{
616
618
  type: Output
617
619
  }] } });
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"]}
620
+ //# 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":"AAAA,OAAO,EACL,SAAS,EACT,KAAK,EACL,MAAM,EACN,YAAY,EACZ,MAAM,EACN,QAAQ,EAGR,SAAS,EACT,MAAM,EACN,iBAAiB,GAClB,MAAM,eAAe,CAAC;AACvB,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;AAC/C,OAAO,EAAE,WAAW,EAAE,MAAM,wBAAwB,CAAC;AACrD,OAAO,EACL,iBAAiB,EACjB,wBAAwB,EACxB,SAAS,EACT,UAAU,EACV,OAAO,EACP,OAAO,EACP,OAAO,EACP,OAAO,GACR,MAAM,2BAA2B,CAAC;AACnC,OAAO,EAAE,cAAc,EAAE,YAAY,EAAE,MAAM,MAAM,CAAC;AACpD,OAAO,EAML,8BAA8B,GAC/B,MAAM,SAAS,CAAC;AAEjB,OAAO,EAAE,kBAAkB,EAAE,MAAM,+CAA+C,CAAC;AACnF,OAAO,EAAE,eAAe,EAAE,MAAM,6CAA6C,CAAC;;;AAG9E;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAqCG;AA2OH,MAAM,OAAO,qBAAqB;IA1OlC;QA2OmB,oBAAe,GAAG,MAAM,CAAC,eAAe,CAAC,CAAC;QAC1C,QAAG,GAAG,MAAM,CAAC,iBAAiB,CAAC,CAAC;QAC9B,SAAI,GAAG,MAAM,CAAC,WAAW,CAAC,CAAC;QAO9C,iBAAiB;QACP,aAAQ,GAAG,IAAI,YAAY,EAAiB,CAAC;QAC7C,YAAO,GAAG,IAAI,YAAY,EAAgB,CAAC;QAC3C,gBAAW,GAAG,IAAI,YAAY,EAAqB,CAAC;QACpD,gBAAW,GAAG,IAAI,YAAY,EAAO,CAAC;QACtC,kBAAa,GAAG,IAAI,YAAY,EAAS,CAAC;QAEpD,yBAAyB;QAChB,UAAK,GAAG,MAAM,CAAM,EAAE,CAAC,CAAC;QACxB,UAAK,GAAG,MAAM,CAAoB,MAAM,CAAC,CAAC;QAC1C,kBAAa,GAAG,MAAM,CAAU,IAAI,CAAC,CAAC;QACtC,eAAU,GAAG,MAAM,CAAU,KAAK,CAAC,CAAC;QACpC,UAAK,GAAG,MAAM,CAAe,IAAI,CAAC,CAAC;QACnC,kBAAa,GAAG,MAAM,CAAU,IAAI,CAAC,CAAC;QAEvC,gBAAW,GAAG,CAAC,CAAC;QAChB,kBAAa,GAAY,IAAI,CAAC;QAEtC,qDAAqD;QAC5C,iBAAY,GAAG,QAAQ,CAAC,GAAG,EAAE;YACpC,IAAI,CAAC,IAAI,CAAC,KAAK,EAAE,UAAU,EAAE,UAAU;gBAAE,OAAO,IAAI,CAAC;YACrD,OAAO,IAAI,CAAC,KAAK,EAAE,CAAC,MAAM,GAAG,IAAI,CAAC,KAAK,CAAC,UAAU,CAAC,UAAU,CAAC;QAChE,CAAC,CAAC,CAAC;QA8BH,kDAAkD;QACzC,uBAAkB,GAAG,QAAQ,CAAC,GAAG,EAAE;YAC1C,QAAQ,IAAI,CAAC,KAAK,EAAE,EAAE,CAAC;gBACrB,KAAK,SAAS;oBACZ,OAAO,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,cAAc,CAAC,CAAC;gBACrC,KAAK,OAAO;oBACV,OAAO,GAAG,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,OAAO,CAAC,KAAK,IAAI,CAAC,KAAK,EAAE,EAAE,OAAO,IAAI,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,eAAe,CAAC,EAAE,CAAC;gBAC7F,KAAK,UAAU;oBACb,OAAO,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,gBAAgB,CAAC,CAAC;gBACvC;oBACE,OAAO,EAAE,CAAC;YACd,CAAC;QACH,CAAC,CAAC,CAAC;KAwNJ;IAhQC,oCAAoC;IACpC,IAAI,WAAW;QACb,OAAO,EAAE,GAAG,8BAA8B,EAAE,GAAG,IAAI,CAAC,KAAK,EAA6B,CAAC;IACzF,CAAC;IAED,2BAA2B;IAC3B,IAAI,eAAe;QACjB,OAAO,IAAI,CAAC,WAAW,CAAC,aAAa,IAAI,EAAE,CAAC;IAC9C,CAAC;IAED,oCAAoC;IACpC,IAAI,iBAAiB;QACnB,MAAM,YAAY,GAAG,IAAI,CAAC,WAAW,CAAC,QAAQ,EAAE,QAAQ,IAAI,MAAM,CAAC;QACnE,MAAM,QAAQ,GAAG,IAAI,CAAC,eAAe,CAAC,WAAW,CAAC,YAAY,CAAC,CAAC;QAChE,OAAO,QAAQ,EAAE,SAAS,IAAI,IAAI,CAAC;IACrC,CAAC;IAED,8BAA8B;IAC9B,IAAI,cAAc;QAChB,OAAO;YACL,MAAM,EAAE;gBACN,KAAK,EAAE,IAAI,CAAC,WAAW,CAAC,QAAQ,EAAE,KAAK,IAAI,CAAC;gBAC5C,QAAQ,EAAE,IAAI;gBACd,GAAG,IAAI,CAAC,WAAW,CAAC,QAAQ,EAAE,MAAM;aACrC;SACF,CAAC;IACJ,CAAC;IAgBD,QAAQ;QACN,mDAAmD;QACnD,IAAI,IAAI,CAAC,KAAK,CAAC,UAAU,CAAC,KAAK,EAAE,MAAM,EAAE,CAAC;YACxC,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,GAAG,IAAI,CAAC,KAAK,CAAC,UAAU,CAAC,KAAK,CAAC,CAAC,CAAC;YACjD,IAAI,CAAC,aAAa,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;QAChC,CAAC;QAED,iCAAiC;QACjC,IAAI,IAAI,CAAC,WAAW,CAAC,QAAQ,IAAI,CAAC,IAAI,CAAC,KAAK,EAAE,CAAC,MAAM,EAAE,CAAC;YACtD,IAAI,CAAC,WAAW,EAAE,CAAC;QACrB,CAAC;IACH,CAAC;IAED,WAAW;QACT,UAAU;IACZ,CAAC;IAED,qCAAqC;IACrC,SAAS,CAAC,KAAa,EAAE,IAAO;QAC9B,IAAI,IAAI,CAAC,KAAK,CAAC,UAAU,CAAC,OAAO,EAAE,CAAC;YAClC,OAAO,IAAI,CAAC,KAAK,CAAC,UAAU,CAAC,OAAO,CAAC,KAAK,EAAE,IAAI,CAAC,CAAC;QACpD,CAAC;QACD,OAAO,KAAK,CAAC;IACf,CAAC;IAED,yCAAyC;IACzC,sBAAsB;QACpB,MAAM,GAAG,GAAG,IAAI,CAAC,WAAW,CAAC,SAAS,CAAC;QACvC,OAAO,CAAC,GAAG,KAAK,QAAQ,IAAI,GAAG,KAAK,MAAM,CAAC,IAAI,IAAI,CAAC,KAAK,EAAE,CAAC,MAAM,GAAG,CAAC,CAAC;IACzE,CAAC;IAED,6BAA6B;IAC7B,KAAK,CAAC,WAAW;QACf,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,UAAU,CAAC,MAAM;YAAE,OAAO;QAE1C,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;QAC1B,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;QACjC,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QAErB,IAAI,CAAC;YACH,MAAM,MAAM,GAAe;gBACzB,SAAS,EAAE,QAAQ;gBACnB,IAAI,EAAE,CAAC;gBACP,QAAQ,EAAE,IAAI,CAAC,WAAW,CAAC,QAAQ,IAAI,EAAE;aAC1C,CAAC;YAEF,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,WAAW,CAAC,MAAM,CAAC,CAAC;YAE9C,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,MAAM,CAAC,KAAY,CAAC,CAAC;YACpC,IAAI,CAAC,aAAa,CAAC,GAAG,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;YACvC,IAAI,CAAC,WAAW,GAAG,CAAC,CAAC;YACrB,IAAI,CAAC,aAAa,GAAG,MAAM,CAAC,MAAM,CAAC;YACnC,IAAI,CAAC,aAAa,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;YAE9B,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC;YACvB,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;YAC9B,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,EAAE,CAAC,CAAC;QACtC,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,IAAI,CAAC,WAAW,CAAC,GAAY,CAAC,CAAC;QACjC,CAAC;IACH,CAAC;IAED,4CAA4C;IAC5C,KAAK,CAAC,UAAU;QACd,IAAI,CAAC,IAAI,CAAC,aAAa,EAAE,IAAI,IAAI,CAAC,KAAK,EAAE,KAAK,SAAS;YAAE,OAAO;QAChE,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,UAAU,CAAC,MAAM;YAAE,OAAO;QAE1C,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;QAC1B,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;QAEjC,IAAI,CAAC;YACH,MAAM,MAAM,GAAe;gBACzB,SAAS,EAAE,QAAQ;gBACnB,IAAI,EAAE,IAAI,CAAC,WAAW;gBACtB,QAAQ,EAAE,IAAI,CAAC,WAAW,CAAC,QAAQ,IAAI,EAAE;gBACzC,MAAM,EAAE,IAAI,CAAC,aAAa;gBAC1B,QAAQ,EAAE,IAAI,CAAC,KAAK,EAAE,CAAC,IAAI,CAAC,KAAK,EAAE,CAAC,MAAM,GAAG,CAAC,CAAC;aAChD,CAAC;YAEF,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,WAAW,CAAC,MAAM,CAAC,CAAC;YAE9C,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC,OAAO,EAAE,EAAE,CAAC,CAAC,GAAG,OAAO,EAAE,GAAI,MAAM,CAAC,KAAa,CAAC,CAAC,CAAC;YACvE,IAAI,CAAC,aAAa,CAAC,GAAG,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;YACvC,IAAI,CAAC,WAAW,EAAE,CAAC;YACnB,IAAI,CAAC,aAAa,GAAG,MAAM,CAAC,MAAM,CAAC;YAEnC,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,UAAU,CAAC,CAAC;YACrD,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,UAAU,CAAC,CAAC;YAC5D,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,EAAE,CAAC,CAAC;QACtC,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,IAAI,CAAC,WAAW,CAAC,GAAY,CAAC,CAAC;QACjC,CAAC;IACH,CAAC;IAED,4CAA4C;IAC5C,KAAK,CAAC,OAAO;QACX,IAAI,CAAC,IAAI,CAAC,UAAU,EAAE,IAAI,IAAI,CAAC,KAAK,EAAE,KAAK,SAAS;YAAE,OAAO;QAC7D,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,UAAU,CAAC,MAAM;YAAE,OAAO;QAE1C,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;QAE1B,IAAI,CAAC;YACH,MAAM,MAAM,GAAe;gBACzB,SAAS,EAAE,KAAK;gBAChB,IAAI,EAAE,CAAC;gBACP,QAAQ,EAAE,IAAI,CAAC,WAAW,CAAC,QAAQ,IAAI,EAAE;gBACzC,SAAS,EAAE,IAAI,CAAC,KAAK,EAAE,CAAC,CAAC,CAAC;aAC3B,CAAC;YAEF,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,WAAW,CAAC,MAAM,CAAC,CAAC;YAE9C,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC,OAAO,EAAE,EAAE,CAAC,CAAC,GAAI,MAAM,CAAC,KAAa,EAAE,GAAG,OAAO,CAAC,CAAC,CAAC;YACvE,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;YAEpC,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC;YACvB,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,EAAE,CAAC,CAAC;QACtC,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,IAAI,CAAC,WAAW,CAAC,GAAY,CAAC,CAAC;QACjC,CAAC;IACH,CAAC;IAED,oCAAoC;IACpC,KAAK,CAAC,WAAW;QACf,IAAI,CAAC,WAAW,GAAG,CAAC,CAAC;QACrB,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC;QAC1B,IAAI,CAAC,aAAa,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QAC7B,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;QACnB,MAAM,IAAI,CAAC,WAAW,EAAE,CAAC;IAC3B,CAAC;IAED,kCAAkC;IAClC,KAAK,CAAC,KAAK;QACT,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QACrB,IAAI,IAAI,CAAC,KAAK,EAAE,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YAC9B,MAAM,IAAI,CAAC,WAAW,EAAE,CAAC;QAC3B,CAAC;aAAM,CAAC;YACN,MAAM,IAAI,CAAC,UAAU,EAAE,CAAC;QAC1B,CAAC;IACH,CAAC;IAED,qBAAqB;IACrB,KAAK,CAAC,KAAK;QACT,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;QACnB,IAAI,CAAC,WAAW,GAAG,CAAC,CAAC;QACrB,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC;QAC1B,IAAI,CAAC,aAAa,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QAC7B,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;QAC3B,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QACrB,IAAI,CAAC,aAAa,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QAC7B,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC;QAEvB,IAAI,IAAI,CAAC,WAAW,CAAC,QAAQ,EAAE,CAAC;YAC9B,MAAM,IAAI,CAAC,WAAW,EAAE,CAAC;QAC3B,CAAC;IACH,CAAC;IAED,8BAA8B;IAC9B,YAAY,CAAC,QAAa;QACxB,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC,OAAO,EAAE,EAAE,CAAC,CAAC,GAAG,QAAQ,EAAE,GAAG,OAAO,CAAC,CAAC,CAAC;QAC1D,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,EAAE,CAAC,CAAC;IACtC,CAAC;IAED,6BAA6B;IAC7B,WAAW,CAAC,QAAa;QACvB,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC,OAAO,EAAE,EAAE,CAAC,CAAC,GAAG,OAAO,EAAE,GAAG,QAAQ,CAAC,CAAC,CAAC;QAC1D,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,EAAE,CAAC,CAAC;IACtC,CAAC;IAED,oCAAoC;IACpC,UAAU,CAAC,KAAa,EAAE,IAAO;QAC/B,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC,OAAO,EAAE,EAAE;YAC5B,MAAM,OAAO,GAAG,CAAC,GAAG,OAAO,CAAC,CAAC;YAC7B,OAAO,CAAC,KAAK,CAAC,GAAG,IAAI,CAAC;YACtB,OAAO,OAAO,CAAC;QACjB,CAAC,CAAC,CAAC;QACH,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,EAAE,CAAC,CAAC;IACtC,CAAC;IAED,iCAAiC;IACjC,UAAU,CAAC,KAAa;QACtB,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC,OAAO,EAAE,EAAE,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,KAAK,KAAK,CAAC,CAAC,CAAC;QACtE,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,EAAE,CAAC,CAAC;IACtC,CAAC;IAED,6CAA6C;IAC7C,KAAK,CAAC,gBAAgB,CAAC,KAAkB;QACvC,MAAM,IAAI,CAAC,UAAU,EAAE,CAAC;QACvB,KAAK,CAAC,MAAuC,CAAC,QAAQ,EAAE,CAAC;IAC5D,CAAC;IAED,qCAAqC;IACrC,KAAK,CAAC,kBAAkB,CAAC,KAAmB;QAC1C,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QACzB,MAAM,IAAI,CAAC,WAAW,EAAE,CAAC;QACzB,KAAK,CAAC,QAAQ,EAAE,CAAC;IACnB,CAAC;IAEO,KAAK,CAAC,WAAW,CAAC,MAAkB;QAC1C,MAAM,MAAM,GAAG,IAAI,CAAC,KAAK,CAAC,UAAU,CAAC,MAAO,CAAC;QAC7C,MAAM,MAAM,GAAG,MAAM,CAAC,MAAM,CAAC,CAAC;QAE9B,IAAI,YAAY,CAAC,MAAM,CAAC,EAAE,CAAC;YACzB,OAAO,MAAM,cAAc,CAAC,MAAM,CAAC,CAAC;QACtC,CAAC;QACD,OAAO,MAAM,MAAM,CAAC;IACtB,CAAC;IAEO,WAAW,CAAC,GAAU;QAC5B,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;QACpB,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC;QACxB,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;QAC/B,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;QAC7B,OAAO,CAAC,KAAK,CAAC,qCAAqC,EAAE,GAAG,CAAC,CAAC;IAC5D,CAAC;+GAjSU,qBAAqB;mGAArB,qBAAqB,wSAKrB,iBAAiB,gDAhOlB;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA+IT,4/BA1JC,YAAY,8cACZ,iBAAiB,+GACjB,wBAAwB,mHACxB,SAAS,oPACT,UAAU,yGACV,OAAO,2JACP,OAAO,gFAGP,kBAAkB;;4FA6NT,qBAAqB;kBA1OjC,SAAS;+BACE,mBAAmB,cACjB,IAAI,WACP;wBACP,YAAY;wBACZ,iBAAiB;wBACjB,wBAAwB;wBACxB,SAAS;wBACT,UAAU;wBACV,OAAO;wBACP,OAAO;wBACP,OAAO;wBACP,OAAO;wBACP,kBAAkB;qBACnB,YACS;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA+IT;8BAiF6B,cAAc;sBAA3C,SAAS;uBAAC,iBAAiB;gBAGnB,KAAK;sBAAb,KAAK;gBAGI,QAAQ;sBAAjB,MAAM;gBACG,OAAO;sBAAhB,MAAM;gBACG,WAAW;sBAApB,MAAM;gBACG,WAAW;sBAApB,MAAM;gBACG,aAAa;sBAAtB,MAAM","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 { I18nService } from '../../../services/i18n';\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 || i18n.t('error') }}</h3>\n              <p>{{ mergedProps.errorState?.message || error()?.message || i18n.t('errorOccurred') }}</p>\n              @if (mergedProps.errorState?.showRetry !== false) {\n                <ion-button fill=\"outline\" (click)=\"retry()\">\n                  {{ mergedProps.errorState?.retryText || i18n.t('retry') }}\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 || i18n.t('loadMore') }}\n                </ion-button>\n              } @else {\n                <ion-text color=\"medium\">{{ mergedProps.noMoreText || i18n.t('noMoreItems') }}</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' ? i18n.t('loading') : ''\"\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 || i18n.t('noMoreItems') }}</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  protected readonly i18n = inject(I18nService);\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 this.i18n.t('loadingItems');\n      case 'error':\n        return `${this.i18n.t('error')}: ${this.error()?.message || this.i18n.t('errorOccurred')}`;\n      case 'complete':\n        return this.i18n.t('allItemsLoaded');\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"]}