valtech-components 2.0.538 → 2.0.540

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.
@@ -5,7 +5,7 @@ import { I18nService } from '../../../services/i18n';
5
5
  import { addIcons } from 'ionicons';
6
6
  import { personOutline, mailOutline, callOutline, ticketOutline, locationOutline, calendarOutline, trophyOutline, ellipsisVertical, checkmarkCircle, timeOutline, closeCircle, } from 'ionicons/icons';
7
7
  import { AvatarComponent } from '../../atoms/avatar/avatar.component';
8
- import { DEFAULT_PAYMENT_STATUS_LABELS, DEFAULT_PAYMENT_STATUS_COLORS, } from './types';
8
+ import { DEFAULT_PAYMENT_STATUS_COLORS, } from './types';
9
9
  import * as i0 from "@angular/core";
10
10
  addIcons({
11
11
  personOutline,
@@ -144,8 +144,20 @@ export class ParticipantCardComponent {
144
144
  const status = this.props.participant.paymentStatus;
145
145
  if (!status)
146
146
  return '';
147
- const labels = { ...DEFAULT_PAYMENT_STATUS_LABELS, ...this.props.paymentStatusLabels };
148
- return labels[status] || status;
147
+ // If custom labels are provided, use them first
148
+ if (this.props.paymentStatusLabels?.[status]) {
149
+ return this.props.paymentStatusLabels[status];
150
+ }
151
+ // Use i18n for default labels
152
+ const i18nKeys = {
153
+ pending: 'paymentPending',
154
+ processing: 'paymentProcessing',
155
+ paid: 'paymentPaid',
156
+ refunded: 'paymentRefunded',
157
+ failed: 'paymentFailed',
158
+ };
159
+ const key = i18nKeys[status];
160
+ return key ? this.i18n.t(key) : status;
149
161
  }
150
162
  getPaymentStatusColor() {
151
163
  const status = this.props.participant.paymentStatus;
@@ -499,4 +511,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
499
511
  }], actionClick: [{
500
512
  type: Output
501
513
  }] } });
502
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"participant-card.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/molecules/participant-card/participant-card.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport { Component, Input, Output, EventEmitter, inject } from '@angular/core';\nimport { IonIcon, IonBadge, IonButton, IonChip } from '@ionic/angular/standalone';\nimport { I18nService } from '../../../services/i18n';\nimport { addIcons } from 'ionicons';\nimport {\n  personOutline,\n  mailOutline,\n  callOutline,\n  ticketOutline,\n  locationOutline,\n  calendarOutline,\n  trophyOutline,\n  ellipsisVertical,\n  checkmarkCircle,\n  timeOutline,\n  closeCircle,\n} from 'ionicons/icons';\nimport { AvatarComponent } from '../../atoms/avatar/avatar.component';\nimport {\n  ParticipantCardMetadata,\n  ParticipantCardClickEvent,\n  ParticipantCardActionEvent,\n  ParticipantTicket,\n  DEFAULT_PAYMENT_STATUS_LABELS,\n  DEFAULT_PAYMENT_STATUS_COLORS,\n} from './types';\n\naddIcons({\n  personOutline,\n  mailOutline,\n  callOutline,\n  ticketOutline,\n  locationOutline,\n  calendarOutline,\n  trophyOutline,\n  ellipsisVertical,\n  checkmarkCircle,\n  timeOutline,\n  closeCircle,\n});\n\n@Component({\n  selector: 'val-participant-card',\n  standalone: true,\n  imports: [CommonModule, IonIcon, IonBadge, IonButton, IonChip, AvatarComponent],\n  template: `\n    <article\n      class=\"participant-card\"\n      [class]=\"props.cssClass\"\n      [class.variant-default]=\"props.variant === 'default' || !props.variant\"\n      [class.variant-compact]=\"props.variant === 'compact'\"\n      [class.variant-detailed]=\"props.variant === 'detailed'\"\n      [class.variant-admin]=\"props.variant === 'admin'\"\n      [class.size-small]=\"props.size === 'small'\"\n      [class.size-medium]=\"props.size === 'medium' || !props.size\"\n      [class.size-large]=\"props.size === 'large'\"\n      [class.is-winner]=\"props.participant.isWinner && props.highlightWinner !== false\"\n      [class.clickable]=\"props.clickable\"\n      (click)=\"onCardClick()\"\n    >\n      <!-- Winner badge -->\n      @if (props.participant.isWinner && props.highlightWinner !== false) {\n        <div class=\"winner-badge\">\n          <ion-icon name=\"trophy-outline\"></ion-icon>\n          <span>{{ getWinnerText() }}</span>\n        </div>\n      }\n\n      <!-- Main content -->\n      <div class=\"card-main\">\n        <!-- Avatar -->\n        @if (props.showAvatar !== false) {\n          <div class=\"avatar-section\">\n            @if (props.participant.avatar) {\n              <val-avatar [props]=\"props.participant.avatar\"></val-avatar>\n            } @else {\n              <div class=\"default-avatar\">\n                <ion-icon name=\"person-outline\"></ion-icon>\n              </div>\n            }\n          </div>\n        }\n\n        <!-- Info -->\n        <div class=\"info-section\">\n          <h4 class=\"participant-name\">{{ props.participant.name }}</h4>\n\n          @if (props.showEmail !== false && props.participant.email) {\n            <div class=\"info-item\">\n              <ion-icon name=\"mail-outline\"></ion-icon>\n              <span>{{ props.participant.email }}</span>\n            </div>\n          }\n\n          @if (props.showPhone !== false && props.participant.phone) {\n            <div class=\"info-item\">\n              <ion-icon name=\"call-outline\"></ion-icon>\n              <span>{{ props.participant.phone }}</span>\n            </div>\n          }\n\n          @if (props.participant.location) {\n            <div class=\"info-item\">\n              <ion-icon name=\"location-outline\"></ion-icon>\n              <span>{{ props.participant.location }}</span>\n            </div>\n          }\n\n          @if (props.showDate !== false && props.participant.registrationDate) {\n            <div class=\"info-item\">\n              <ion-icon name=\"calendar-outline\"></ion-icon>\n              <span>{{ formatDate(props.participant.registrationDate) }}</span>\n            </div>\n          }\n        </div>\n\n        <!-- Status & Amount (right side) -->\n        <div class=\"status-section\">\n          @if (props.showPaymentStatus !== false && props.participant.paymentStatus) {\n            <ion-badge [color]=\"getPaymentStatusColor()\" class=\"payment-badge\">\n              <ion-icon [name]=\"getPaymentStatusIcon()\"></ion-icon>\n              {{ getPaymentStatusLabel() }}\n            </ion-badge>\n          }\n\n          @if (props.showAmount !== false && props.participant.amountPaid) {\n            <div class=\"amount\">\n              {{ formatCurrency(props.participant.amountPaid) }}\n            </div>\n          }\n        </div>\n\n        <!-- Actions menu -->\n        @if (props.showActions && props.actions?.length) {\n          <ion-button\n            fill=\"clear\"\n            class=\"actions-button\"\n            (click)=\"onActionsClick($event)\"\n          >\n            <ion-icon slot=\"icon-only\" name=\"ellipsis-vertical\"></ion-icon>\n          </ion-button>\n        }\n      </div>\n\n      <!-- Tickets section -->\n      @if (props.showTickets !== false && ticketNumbers.length > 0) {\n        <div class=\"tickets-section\">\n          <div class=\"tickets-header\">\n            <ion-icon name=\"ticket-outline\"></ion-icon>\n            <span>{{ getTicketsCountText(ticketNumbers.length) }}</span>\n          </div>\n\n          <div class=\"tickets-list\">\n            @for (ticket of visibleTickets; track $index) {\n              <ion-chip\n                [color]=\"getTicketColor(ticket)\"\n                [class.winner-ticket]=\"isWinnerTicket(ticket)\"\n                class=\"ticket-chip\"\n              >\n                @if (isWinnerTicket(ticket)) {\n                  <ion-icon name=\"trophy-outline\"></ion-icon>\n                }\n                {{ formatTicketNumber(ticket) }}\n              </ion-chip>\n            }\n\n            @if (hiddenTicketsCount > 0) {\n              <ion-chip\n                color=\"medium\"\n                class=\"more-tickets\"\n                (click)=\"toggleShowAllTickets($event)\"\n              >\n                {{ getMoreText(hiddenTicketsCount) }}\n              </ion-chip>\n            }\n          </div>\n        </div>\n      }\n\n      <!-- Notes (admin variant) -->\n      @if (props.variant === 'admin' && props.participant.notes) {\n        <div class=\"notes-section\">\n          <span class=\"notes-label\">{{ getNotesLabel() }}</span>\n          <p class=\"notes-text\">{{ props.participant.notes }}</p>\n        </div>\n      }\n    </article>\n  `,\n  styleUrls: ['./participant-card.component.scss'],\n})\n/**\n * val-participant-card\n *\n * A card component for displaying participant information in a raffle.\n *\n * @example Basic usage\n * ```html\n * <val-participant-card\n *   [props]=\"{\n *     participant: {\n *       id: '1',\n *       name: 'Juan Pérez',\n *       email: 'juan@email.com',\n *       tickets: [42, 156, 789],\n *       paymentStatus: 'paid'\n *     }\n *   }\"\n * ></val-participant-card>\n * ```\n *\n * @example Admin variant with actions\n * ```html\n * <val-participant-card\n *   [props]=\"{\n *     participant: participant,\n *     variant: 'admin',\n *     showActions: true,\n *     actions: [\n *       { id: 'edit', label: 'Editar', icon: 'create-outline' },\n *       { id: 'refund', label: 'Reembolsar', icon: 'cash-outline', color: 'warning' },\n *       { id: 'delete', label: 'Eliminar', icon: 'trash-outline', destructive: true }\n *     ]\n *   }\"\n *   (cardClick)=\"onParticipantClick($event)\"\n *   (actionClick)=\"onActionClick($event)\"\n * ></val-participant-card>\n * ```\n *\n * @example Winner highlight\n * ```html\n * <val-participant-card\n *   [props]=\"{\n *     participant: {\n *       id: '1',\n *       name: 'María García',\n *       tickets: [{ number: 42, status: 'winner' }],\n *       isWinner: true,\n *       paymentStatus: 'paid'\n *     },\n *     highlightWinner: true,\n *     variant: 'detailed'\n *   }\"\n * ></val-participant-card>\n * ```\n */\nexport class ParticipantCardComponent {\n  @Input() props: ParticipantCardMetadata;\n\n  @Output() cardClick = new EventEmitter<ParticipantCardClickEvent>();\n  @Output() actionClick = new EventEmitter<ParticipantCardActionEvent>();\n\n  private i18n = inject(I18nService);\n\n  showAllTickets = false;\n\n  /** Get winner badge text */\n  getWinnerText(): string {\n    return this.i18n.t('winner');\n  }\n\n  /** Get tickets count text */\n  getTicketsCountText(count: number): string {\n    return `${count} ${count !== 1 ? this.i18n.t('tickets') : this.i18n.t('ticket')}`;\n  }\n\n  /** Get more tickets text */\n  getMoreText(count: number): string {\n    return `+${count} ${this.i18n.t('more')}`;\n  }\n\n  /** Get notes label */\n  getNotesLabel(): string {\n    return this.i18n.t('notes');\n  }\n\n  get ticketNumbers(): (number | ParticipantTicket)[] {\n    return this.props.participant.tickets || [];\n  }\n\n  get visibleTickets(): (number | ParticipantTicket)[] {\n    const max = this.props.maxTicketsVisible || 5;\n    if (this.showAllTickets || this.ticketNumbers.length <= max) {\n      return this.ticketNumbers;\n    }\n    return this.ticketNumbers.slice(0, max);\n  }\n\n  get hiddenTicketsCount(): number {\n    if (this.showAllTickets) return 0;\n    const max = this.props.maxTicketsVisible || 5;\n    return Math.max(0, this.ticketNumbers.length - max);\n  }\n\n  formatTicketNumber(ticket: number | ParticipantTicket): string {\n    const num = typeof ticket === 'number' ? ticket : ticket.number;\n    if (this.props.numberPadding) {\n      return String(num).padStart(this.props.numberPadding, '0');\n    }\n    return String(num);\n  }\n\n  getTicketColor(ticket: number | ParticipantTicket): string {\n    if (typeof ticket === 'number') return 'primary';\n    switch (ticket.status) {\n      case 'winner':\n        return 'warning';\n      case 'reserved':\n        return 'tertiary';\n      case 'cancelled':\n        return 'medium';\n      default:\n        return 'primary';\n    }\n  }\n\n  isWinnerTicket(ticket: number | ParticipantTicket): boolean {\n    if (typeof ticket === 'number') return false;\n    return ticket.status === 'winner';\n  }\n\n  getPaymentStatusLabel(): string {\n    const status = this.props.participant.paymentStatus;\n    if (!status) return '';\n    const labels = { ...DEFAULT_PAYMENT_STATUS_LABELS, ...this.props.paymentStatusLabels };\n    return labels[status] || status;\n  }\n\n  getPaymentStatusColor(): string {\n    const status = this.props.participant.paymentStatus;\n    if (!status) return 'medium';\n    return DEFAULT_PAYMENT_STATUS_COLORS[status] || 'medium';\n  }\n\n  getPaymentStatusIcon(): string {\n    const status = this.props.participant.paymentStatus;\n    switch (status) {\n      case 'paid':\n        return 'checkmark-circle';\n      case 'pending':\n      case 'processing':\n        return 'time-outline';\n      case 'failed':\n      case 'refunded':\n        return 'close-circle';\n      default:\n        return 'time-outline';\n    }\n  }\n\n  formatCurrency(amount: number): string {\n    const symbol = this.props.currencySymbol || '$';\n    return `${symbol}${amount.toLocaleString('es-MX')}`;\n  }\n\n  formatDate(date: Date | string): string {\n    const d = typeof date === 'string' ? new Date(date) : date;\n    return d.toLocaleDateString('es-MX', {\n      day: 'numeric',\n      month: 'short',\n      year: 'numeric',\n    });\n  }\n\n  toggleShowAllTickets(event: Event): void {\n    event.stopPropagation();\n    this.showAllTickets = !this.showAllTickets;\n  }\n\n  onCardClick(): void {\n    if (!this.props.clickable) return;\n    this.cardClick.emit({\n      participant: this.props.participant,\n    });\n  }\n\n  onActionsClick(event: Event): void {\n    event.stopPropagation();\n    // In a real implementation, this would open a popover menu\n    // For now, we'll emit the first action\n    if (this.props.actions?.length) {\n      this.actionClick.emit({\n        actionId: this.props.actions[0].id,\n        participant: this.props.participant,\n      });\n    }\n  }\n}\n"]}
514
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"participant-card.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/molecules/participant-card/participant-card.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport { Component, Input, Output, EventEmitter, inject } from '@angular/core';\nimport { IonIcon, IonBadge, IonButton, IonChip } from '@ionic/angular/standalone';\nimport { I18nService } from '../../../services/i18n';\nimport { addIcons } from 'ionicons';\nimport {\n  personOutline,\n  mailOutline,\n  callOutline,\n  ticketOutline,\n  locationOutline,\n  calendarOutline,\n  trophyOutline,\n  ellipsisVertical,\n  checkmarkCircle,\n  timeOutline,\n  closeCircle,\n} from 'ionicons/icons';\nimport { AvatarComponent } from '../../atoms/avatar/avatar.component';\nimport {\n  ParticipantCardMetadata,\n  ParticipantCardClickEvent,\n  ParticipantCardActionEvent,\n  ParticipantTicket,\n  DEFAULT_PAYMENT_STATUS_COLORS,\n} from './types';\n\naddIcons({\n  personOutline,\n  mailOutline,\n  callOutline,\n  ticketOutline,\n  locationOutline,\n  calendarOutline,\n  trophyOutline,\n  ellipsisVertical,\n  checkmarkCircle,\n  timeOutline,\n  closeCircle,\n});\n\n@Component({\n  selector: 'val-participant-card',\n  standalone: true,\n  imports: [CommonModule, IonIcon, IonBadge, IonButton, IonChip, AvatarComponent],\n  template: `\n    <article\n      class=\"participant-card\"\n      [class]=\"props.cssClass\"\n      [class.variant-default]=\"props.variant === 'default' || !props.variant\"\n      [class.variant-compact]=\"props.variant === 'compact'\"\n      [class.variant-detailed]=\"props.variant === 'detailed'\"\n      [class.variant-admin]=\"props.variant === 'admin'\"\n      [class.size-small]=\"props.size === 'small'\"\n      [class.size-medium]=\"props.size === 'medium' || !props.size\"\n      [class.size-large]=\"props.size === 'large'\"\n      [class.is-winner]=\"props.participant.isWinner && props.highlightWinner !== false\"\n      [class.clickable]=\"props.clickable\"\n      (click)=\"onCardClick()\"\n    >\n      <!-- Winner badge -->\n      @if (props.participant.isWinner && props.highlightWinner !== false) {\n        <div class=\"winner-badge\">\n          <ion-icon name=\"trophy-outline\"></ion-icon>\n          <span>{{ getWinnerText() }}</span>\n        </div>\n      }\n\n      <!-- Main content -->\n      <div class=\"card-main\">\n        <!-- Avatar -->\n        @if (props.showAvatar !== false) {\n          <div class=\"avatar-section\">\n            @if (props.participant.avatar) {\n              <val-avatar [props]=\"props.participant.avatar\"></val-avatar>\n            } @else {\n              <div class=\"default-avatar\">\n                <ion-icon name=\"person-outline\"></ion-icon>\n              </div>\n            }\n          </div>\n        }\n\n        <!-- Info -->\n        <div class=\"info-section\">\n          <h4 class=\"participant-name\">{{ props.participant.name }}</h4>\n\n          @if (props.showEmail !== false && props.participant.email) {\n            <div class=\"info-item\">\n              <ion-icon name=\"mail-outline\"></ion-icon>\n              <span>{{ props.participant.email }}</span>\n            </div>\n          }\n\n          @if (props.showPhone !== false && props.participant.phone) {\n            <div class=\"info-item\">\n              <ion-icon name=\"call-outline\"></ion-icon>\n              <span>{{ props.participant.phone }}</span>\n            </div>\n          }\n\n          @if (props.participant.location) {\n            <div class=\"info-item\">\n              <ion-icon name=\"location-outline\"></ion-icon>\n              <span>{{ props.participant.location }}</span>\n            </div>\n          }\n\n          @if (props.showDate !== false && props.participant.registrationDate) {\n            <div class=\"info-item\">\n              <ion-icon name=\"calendar-outline\"></ion-icon>\n              <span>{{ formatDate(props.participant.registrationDate) }}</span>\n            </div>\n          }\n        </div>\n\n        <!-- Status & Amount (right side) -->\n        <div class=\"status-section\">\n          @if (props.showPaymentStatus !== false && props.participant.paymentStatus) {\n            <ion-badge [color]=\"getPaymentStatusColor()\" class=\"payment-badge\">\n              <ion-icon [name]=\"getPaymentStatusIcon()\"></ion-icon>\n              {{ getPaymentStatusLabel() }}\n            </ion-badge>\n          }\n\n          @if (props.showAmount !== false && props.participant.amountPaid) {\n            <div class=\"amount\">\n              {{ formatCurrency(props.participant.amountPaid) }}\n            </div>\n          }\n        </div>\n\n        <!-- Actions menu -->\n        @if (props.showActions && props.actions?.length) {\n          <ion-button\n            fill=\"clear\"\n            class=\"actions-button\"\n            (click)=\"onActionsClick($event)\"\n          >\n            <ion-icon slot=\"icon-only\" name=\"ellipsis-vertical\"></ion-icon>\n          </ion-button>\n        }\n      </div>\n\n      <!-- Tickets section -->\n      @if (props.showTickets !== false && ticketNumbers.length > 0) {\n        <div class=\"tickets-section\">\n          <div class=\"tickets-header\">\n            <ion-icon name=\"ticket-outline\"></ion-icon>\n            <span>{{ getTicketsCountText(ticketNumbers.length) }}</span>\n          </div>\n\n          <div class=\"tickets-list\">\n            @for (ticket of visibleTickets; track $index) {\n              <ion-chip\n                [color]=\"getTicketColor(ticket)\"\n                [class.winner-ticket]=\"isWinnerTicket(ticket)\"\n                class=\"ticket-chip\"\n              >\n                @if (isWinnerTicket(ticket)) {\n                  <ion-icon name=\"trophy-outline\"></ion-icon>\n                }\n                {{ formatTicketNumber(ticket) }}\n              </ion-chip>\n            }\n\n            @if (hiddenTicketsCount > 0) {\n              <ion-chip\n                color=\"medium\"\n                class=\"more-tickets\"\n                (click)=\"toggleShowAllTickets($event)\"\n              >\n                {{ getMoreText(hiddenTicketsCount) }}\n              </ion-chip>\n            }\n          </div>\n        </div>\n      }\n\n      <!-- Notes (admin variant) -->\n      @if (props.variant === 'admin' && props.participant.notes) {\n        <div class=\"notes-section\">\n          <span class=\"notes-label\">{{ getNotesLabel() }}</span>\n          <p class=\"notes-text\">{{ props.participant.notes }}</p>\n        </div>\n      }\n    </article>\n  `,\n  styleUrls: ['./participant-card.component.scss'],\n})\n/**\n * val-participant-card\n *\n * A card component for displaying participant information in a raffle.\n *\n * @example Basic usage\n * ```html\n * <val-participant-card\n *   [props]=\"{\n *     participant: {\n *       id: '1',\n *       name: 'Juan Pérez',\n *       email: 'juan@email.com',\n *       tickets: [42, 156, 789],\n *       paymentStatus: 'paid'\n *     }\n *   }\"\n * ></val-participant-card>\n * ```\n *\n * @example Admin variant with actions\n * ```html\n * <val-participant-card\n *   [props]=\"{\n *     participant: participant,\n *     variant: 'admin',\n *     showActions: true,\n *     actions: [\n *       { id: 'edit', label: 'Editar', icon: 'create-outline' },\n *       { id: 'refund', label: 'Reembolsar', icon: 'cash-outline', color: 'warning' },\n *       { id: 'delete', label: 'Eliminar', icon: 'trash-outline', destructive: true }\n *     ]\n *   }\"\n *   (cardClick)=\"onParticipantClick($event)\"\n *   (actionClick)=\"onActionClick($event)\"\n * ></val-participant-card>\n * ```\n *\n * @example Winner highlight\n * ```html\n * <val-participant-card\n *   [props]=\"{\n *     participant: {\n *       id: '1',\n *       name: 'María García',\n *       tickets: [{ number: 42, status: 'winner' }],\n *       isWinner: true,\n *       paymentStatus: 'paid'\n *     },\n *     highlightWinner: true,\n *     variant: 'detailed'\n *   }\"\n * ></val-participant-card>\n * ```\n */\nexport class ParticipantCardComponent {\n  @Input() props: ParticipantCardMetadata;\n\n  @Output() cardClick = new EventEmitter<ParticipantCardClickEvent>();\n  @Output() actionClick = new EventEmitter<ParticipantCardActionEvent>();\n\n  private i18n = inject(I18nService);\n\n  showAllTickets = false;\n\n  /** Get winner badge text */\n  getWinnerText(): string {\n    return this.i18n.t('winner');\n  }\n\n  /** Get tickets count text */\n  getTicketsCountText(count: number): string {\n    return `${count} ${count !== 1 ? this.i18n.t('tickets') : this.i18n.t('ticket')}`;\n  }\n\n  /** Get more tickets text */\n  getMoreText(count: number): string {\n    return `+${count} ${this.i18n.t('more')}`;\n  }\n\n  /** Get notes label */\n  getNotesLabel(): string {\n    return this.i18n.t('notes');\n  }\n\n  get ticketNumbers(): (number | ParticipantTicket)[] {\n    return this.props.participant.tickets || [];\n  }\n\n  get visibleTickets(): (number | ParticipantTicket)[] {\n    const max = this.props.maxTicketsVisible || 5;\n    if (this.showAllTickets || this.ticketNumbers.length <= max) {\n      return this.ticketNumbers;\n    }\n    return this.ticketNumbers.slice(0, max);\n  }\n\n  get hiddenTicketsCount(): number {\n    if (this.showAllTickets) return 0;\n    const max = this.props.maxTicketsVisible || 5;\n    return Math.max(0, this.ticketNumbers.length - max);\n  }\n\n  formatTicketNumber(ticket: number | ParticipantTicket): string {\n    const num = typeof ticket === 'number' ? ticket : ticket.number;\n    if (this.props.numberPadding) {\n      return String(num).padStart(this.props.numberPadding, '0');\n    }\n    return String(num);\n  }\n\n  getTicketColor(ticket: number | ParticipantTicket): string {\n    if (typeof ticket === 'number') return 'primary';\n    switch (ticket.status) {\n      case 'winner':\n        return 'warning';\n      case 'reserved':\n        return 'tertiary';\n      case 'cancelled':\n        return 'medium';\n      default:\n        return 'primary';\n    }\n  }\n\n  isWinnerTicket(ticket: number | ParticipantTicket): boolean {\n    if (typeof ticket === 'number') return false;\n    return ticket.status === 'winner';\n  }\n\n  getPaymentStatusLabel(): string {\n    const status = this.props.participant.paymentStatus;\n    if (!status) return '';\n\n    // If custom labels are provided, use them first\n    if (this.props.paymentStatusLabels?.[status]) {\n      return this.props.paymentStatusLabels[status];\n    }\n\n    // Use i18n for default labels\n    const i18nKeys: Record<string, string> = {\n      pending: 'paymentPending',\n      processing: 'paymentProcessing',\n      paid: 'paymentPaid',\n      refunded: 'paymentRefunded',\n      failed: 'paymentFailed',\n    };\n\n    const key = i18nKeys[status];\n    return key ? this.i18n.t(key) : status;\n  }\n\n  getPaymentStatusColor(): string {\n    const status = this.props.participant.paymentStatus;\n    if (!status) return 'medium';\n    return DEFAULT_PAYMENT_STATUS_COLORS[status] || 'medium';\n  }\n\n  getPaymentStatusIcon(): string {\n    const status = this.props.participant.paymentStatus;\n    switch (status) {\n      case 'paid':\n        return 'checkmark-circle';\n      case 'pending':\n      case 'processing':\n        return 'time-outline';\n      case 'failed':\n      case 'refunded':\n        return 'close-circle';\n      default:\n        return 'time-outline';\n    }\n  }\n\n  formatCurrency(amount: number): string {\n    const symbol = this.props.currencySymbol || '$';\n    return `${symbol}${amount.toLocaleString('es-MX')}`;\n  }\n\n  formatDate(date: Date | string): string {\n    const d = typeof date === 'string' ? new Date(date) : date;\n    return d.toLocaleDateString('es-MX', {\n      day: 'numeric',\n      month: 'short',\n      year: 'numeric',\n    });\n  }\n\n  toggleShowAllTickets(event: Event): void {\n    event.stopPropagation();\n    this.showAllTickets = !this.showAllTickets;\n  }\n\n  onCardClick(): void {\n    if (!this.props.clickable) return;\n    this.cardClick.emit({\n      participant: this.props.participant,\n    });\n  }\n\n  onActionsClick(event: Event): void {\n    event.stopPropagation();\n    // In a real implementation, this would open a popover menu\n    // For now, we'll emit the first action\n    if (this.props.actions?.length) {\n      this.actionClick.emit({\n        actionId: this.props.actions[0].id,\n        participant: this.props.participant,\n      });\n    }\n  }\n}\n"]}
@@ -44,6 +44,7 @@ export class DocsTocComponent {
44
44
  this.headingElements = [];
45
45
  this.scrollHandler = null;
46
46
  this.rafId = null;
47
+ this.scrollContainer = null;
47
48
  }
48
49
  ngOnInit() {
49
50
  this.updateFlatItems();
@@ -131,8 +132,12 @@ export class DocsTocComponent {
131
132
  this.headingElements = items
132
133
  .map(item => document.getElementById(item.id))
133
134
  .filter((el) => el !== null);
134
- if (this.headingElements.length === 0)
135
+ if (this.headingElements.length === 0) {
136
+ console.warn('[docs-toc] No heading elements found for IDs:', items.map(i => i.id));
135
137
  return;
138
+ }
139
+ // Find the scroll container - check for ion-content first, then use window
140
+ this.scrollContainer = this.findScrollContainer();
136
141
  // Set initial active item
137
142
  this.updateActiveHeading();
138
143
  // Use scroll event with requestAnimationFrame for performance
@@ -145,23 +150,40 @@ export class DocsTocComponent {
145
150
  this.updateActiveHeading();
146
151
  });
147
152
  };
153
+ // Listen on both window and the scroll container for maximum compatibility
148
154
  window.addEventListener('scroll', this.scrollHandler, { passive: true });
155
+ document.addEventListener('scroll', this.scrollHandler, { passive: true, capture: true });
156
+ // Also listen on ion-content if present
157
+ const ionContent = document.querySelector('ion-content');
158
+ if (ionContent) {
159
+ ionContent.addEventListener('ionScroll', this.scrollHandler, { passive: true });
160
+ }
149
161
  });
150
162
  }
163
+ findScrollContainer() {
164
+ // Check for ion-content's scroll element
165
+ const ionContent = document.querySelector('ion-content');
166
+ if (ionContent) {
167
+ // ion-content has a shadow DOM with the actual scrollable element
168
+ const scrollEl = ionContent.shadowRoot?.querySelector('.inner-scroll');
169
+ if (scrollEl)
170
+ return scrollEl;
171
+ }
172
+ return window;
173
+ }
151
174
  updateActiveHeading() {
152
175
  const offsetTop = this.props.offsetTop ?? 100;
153
- const scrollY = window.scrollY;
154
- // Find the heading that is currently at or above the offset point
176
+ // Find the heading closest to or above the trigger point (offset from top of viewport)
177
+ // Using getBoundingClientRect().top which is relative to viewport - works regardless of scroll container
155
178
  let activeHeading = null;
156
179
  for (const heading of this.headingElements) {
157
180
  const rect = heading.getBoundingClientRect();
158
- const headingTop = rect.top + scrollY;
159
- // If heading is at or above the trigger point (scrollY + offset)
160
- if (headingTop <= scrollY + offsetTop + 10) {
181
+ // If heading's top is at or above the trigger point (offset pixels from viewport top)
182
+ if (rect.top <= offsetTop + 10) {
161
183
  activeHeading = heading;
162
184
  }
163
185
  else {
164
- // Headings are in order, so we can stop once we find one below
186
+ // Since headings are in DOM order, stop once we find one below the trigger
165
187
  break;
166
188
  }
167
189
  }
@@ -179,6 +201,11 @@ export class DocsTocComponent {
179
201
  destroyScrollSpy() {
180
202
  if (this.scrollHandler) {
181
203
  window.removeEventListener('scroll', this.scrollHandler);
204
+ document.removeEventListener('scroll', this.scrollHandler, { capture: true });
205
+ const ionContent = document.querySelector('ion-content');
206
+ if (ionContent) {
207
+ ionContent.removeEventListener('ionScroll', this.scrollHandler);
208
+ }
182
209
  this.scrollHandler = null;
183
210
  }
184
211
  if (this.rafId) {
@@ -186,6 +213,7 @@ export class DocsTocComponent {
186
213
  this.rafId = null;
187
214
  }
188
215
  this.headingElements = [];
216
+ this.scrollContainer = null;
189
217
  }
190
218
  scrollToSection(event, id) {
191
219
  event.preventDefault();
@@ -193,11 +221,22 @@ export class DocsTocComponent {
193
221
  if (!element)
194
222
  return;
195
223
  const offsetTop = this.props.offsetTop ?? 100;
196
- const top = element.getBoundingClientRect().top + window.scrollY - offsetTop;
197
- window.scrollTo({
198
- top,
199
- behavior: 'smooth',
200
- });
224
+ // Use scrollIntoView with offset calculation for better cross-container support
225
+ const elementRect = element.getBoundingClientRect();
226
+ const currentScrollY = window.scrollY || document.documentElement.scrollTop;
227
+ const targetY = elementRect.top + currentScrollY - offsetTop;
228
+ // Try ion-content scrollToPoint first (for Ionic apps)
229
+ const ionContent = document.querySelector('ion-content');
230
+ if (ionContent && typeof ionContent.scrollToPoint === 'function') {
231
+ ionContent.scrollToPoint(0, targetY, 300);
232
+ }
233
+ else {
234
+ // Fallback to window scroll
235
+ window.scrollTo({
236
+ top: targetY,
237
+ behavior: 'smooth',
238
+ });
239
+ }
201
240
  // Update active immediately for better UX
202
241
  this.activeId.set(id);
203
242
  // Update URL hash without jumping
@@ -276,4 +315,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
276
315
  }], sectionChange: [{
277
316
  type: Output
278
317
  }] } });
279
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"docs-toc.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/organisms/docs-toc/docs-toc.component.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;AAC/C,OAAO,EACL,SAAS,EACT,KAAK,EACL,MAAM,EACN,YAAY,EAOZ,MAAM,GACP,MAAM,eAAe,CAAC;;AAGvB;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA+BG;AAuCH,MAAM,OAAO,gBAAgB;IAa3B,YAAoB,MAAc;QAAd,WAAM,GAAN,MAAM,CAAQ;QAZzB,UAAK,GAAoB,EAAE,CAAC;QAE3B,kBAAa,GAAG,IAAI,YAAY,EAAU,CAAC;QAErD,iBAAiB;QACP,aAAQ,GAAG,MAAM,CAAS,EAAE,CAAC,CAAC;QAC9B,cAAS,GAAG,MAAM,CAAgB,EAAE,CAAC,CAAC;QAExC,oBAAe,GAAkB,EAAE,CAAC;QACpC,kBAAa,GAAwB,IAAI,CAAC;QAC1C,UAAK,GAAkB,IAAI,CAAC;IAEC,CAAC;IAEtC,QAAQ;QACN,IAAI,CAAC,eAAe,EAAE,CAAC;IACzB,CAAC;IAED,eAAe;QACb,+BAA+B;QAC/B,UAAU,CAAC,GAAG,EAAE;YACd,IAAI,IAAI,CAAC,KAAK,CAAC,YAAY,EAAE,CAAC;gBAC5B,IAAI,CAAC,uBAAuB,EAAE,CAAC;YACjC,CAAC;YACD,IAAI,CAAC,cAAc,EAAE,CAAC;QACxB,CAAC,EAAE,GAAG,CAAC,CAAC;IACV,CAAC;IAED,WAAW,CAAC,OAAsB;QAChC,IAAI,OAAO,CAAC,OAAO,CAAC,EAAE,CAAC;YACrB,IAAI,CAAC,eAAe,EAAE,CAAC;YACvB,IAAI,IAAI,CAAC,KAAK,CAAC,YAAY,EAAE,CAAC;gBAC5B,UAAU,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,uBAAuB,EAAE,EAAE,GAAG,CAAC,CAAC;YACxD,CAAC;QACH,CAAC;IACH,CAAC;IAED,WAAW;QACT,IAAI,CAAC,gBAAgB,EAAE,CAAC;IAC1B,CAAC;IAEO,eAAe;QACrB,IAAI,IAAI,CAAC,KAAK,CAAC,KAAK,EAAE,CAAC;YACrB,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC,CAAC;QAC1D,CAAC;IACH,CAAC;IAEO,YAAY,CAAC,KAAoB;QACvC,MAAM,MAAM,GAAkB,EAAE,CAAC;QACjC,MAAM,OAAO,GAAG,CAAC,QAAuB,EAAE,EAAE;YAC1C,QAAQ,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE;gBACtB,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;gBAClB,IAAI,IAAI,CAAC,QAAQ,EAAE,MAAM,EAAE,CAAC;oBAC1B,OAAO,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;gBACzB,CAAC;YACH,CAAC,CAAC,CAAC;QACL,CAAC,CAAC;QACF,OAAO,CAAC,KAAK,CAAC,CAAC;QACf,OAAO,MAAM,CAAC;IAChB,CAAC;IAEO,uBAAuB;QAC7B,MAAM,QAAQ,GAAG,IAAI,CAAC,KAAK,CAAC,iBAAiB,IAAI,uBAAuB,CAAC;QACzE,MAAM,SAAS,GAAG,QAAQ,CAAC,aAAa,CAAC,QAAQ,CAAC,CAAC;QAEnD,IAAI,CAAC,SAAS,EAAE,CAAC;YACf,OAAO,CAAC,IAAI,CAAC,mCAAmC,QAAQ,EAAE,CAAC,CAAC;YAC5D,OAAO;QACT,CAAC;QAED,MAAM,MAAM,GAAG,IAAI,CAAC,KAAK,CAAC,aAAa,IAAI,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC;QAClD,MAAM,eAAe,GAAG,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAC5D,MAAM,QAAQ,GAAG,SAAS,CAAC,gBAAgB,CAAC,eAAe,CAAC,CAAC;QAE7D,MAAM,KAAK,GAAkB,EAAE,CAAC;QAChC,QAAQ,CAAC,OAAO,CAAC,CAAC,OAAgB,EAAE,EAAE;YACpC,MAAM,EAAE,GAAG,OAAsB,CAAC;YAClC,6BAA6B;YAC7B,IAAI,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC;gBACX,EAAE,CAAC,EAAE,GAAG,IAAI,CAAC,UAAU,CAAC,EAAE,CAAC,WAAW,IAAI,EAAE,CAAC,CAAC;YAChD,CAAC;YAED,MAAM,KAAK,GAAG,QAAQ,CAAC,EAAE,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;YAC1C,KAAK,CAAC,IAAI,CAAC;gBACT,EAAE,EAAE,EAAE,CAAC,EAAE;gBACT,KAAK,EAAE,EAAE,CAAC,WAAW,EAAE,IAAI,EAAE,IAAI,EAAE;gBACnC,KAAK,EAAE,KAAK,GAAG,CAAC,EAAE,6BAA6B;aAChD,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;QAC1B,qCAAqC;QACrC,IAAI,CAAC,cAAc,EAAE,CAAC;IACxB,CAAC;IAEO,UAAU,CAAC,IAAY;QAC7B,OAAO,IAAI;aACR,WAAW,EAAE;aACb,IAAI,EAAE;aACN,OAAO,CAAC,WAAW,EAAE,EAAE,CAAC;aACxB,OAAO,CAAC,MAAM,EAAE,GAAG,CAAC;aACpB,SAAS,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;IACtB,CAAC;IAEO,cAAc;QACpB,IAAI,CAAC,gBAAgB,EAAE,CAAC;QAExB,MAAM,KAAK,GAAG,IAAI,CAAC,SAAS,EAAE,CAAC;QAC/B,IAAI,KAAK,CAAC,MAAM,KAAK,CAAC;YAAE,OAAO;QAE/B,2BAA2B;QAC3B,IAAI,CAAC,eAAe,GAAG,KAAK;aACzB,GAAG,CAAC,IAAI,CAAC,EAAE,CAAC,QAAQ,CAAC,cAAc,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;aAC7C,MAAM,CAAC,CAAC,EAAE,EAAqB,EAAE,CAAC,EAAE,KAAK,IAAI,CAAC,CAAC;QAElD,IAAI,IAAI,CAAC,eAAe,CAAC,MAAM,KAAK,CAAC;YAAE,OAAO;QAE9C,0BAA0B;QAC1B,IAAI,CAAC,mBAAmB,EAAE,CAAC;QAE3B,8DAA8D;QAC9D,IAAI,CAAC,MAAM,CAAC,iBAAiB,CAAC,GAAG,EAAE;YACjC,IAAI,CAAC,aAAa,GAAG,GAAG,EAAE;gBACxB,IAAI,IAAI,CAAC,KAAK,EAAE,CAAC;oBACf,oBAAoB,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;gBACnC,CAAC;gBACD,IAAI,CAAC,KAAK,GAAG,qBAAqB,CAAC,GAAG,EAAE;oBACtC,IAAI,CAAC,mBAAmB,EAAE,CAAC;gBAC7B,CAAC,CAAC,CAAC;YACL,CAAC,CAAC;YAEF,MAAM,CAAC,gBAAgB,CAAC,QAAQ,EAAE,IAAI,CAAC,aAAa,EAAE,EAAE,OAAO,EAAE,IAAI,EAAE,CAAC,CAAC;QAC3E,CAAC,CAAC,CAAC;IACL,CAAC;IAEO,mBAAmB;QACzB,MAAM,SAAS,GAAG,IAAI,CAAC,KAAK,CAAC,SAAS,IAAI,GAAG,CAAC;QAC9C,MAAM,OAAO,GAAG,MAAM,CAAC,OAAO,CAAC;QAE/B,kEAAkE;QAClE,IAAI,aAAa,GAAuB,IAAI,CAAC;QAE7C,KAAK,MAAM,OAAO,IAAI,IAAI,CAAC,eAAe,EAAE,CAAC;YAC3C,MAAM,IAAI,GAAG,OAAO,CAAC,qBAAqB,EAAE,CAAC;YAC7C,MAAM,UAAU,GAAG,IAAI,CAAC,GAAG,GAAG,OAAO,CAAC;YAEtC,iEAAiE;YACjE,IAAI,UAAU,IAAI,OAAO,GAAG,SAAS,GAAG,EAAE,EAAE,CAAC;gBAC3C,aAAa,GAAG,OAAO,CAAC;YAC1B,CAAC;iBAAM,CAAC;gBACN,+DAA+D;gBAC/D,MAAM;YACR,CAAC;QACH,CAAC;QAED,8DAA8D;QAC9D,IAAI,CAAC,aAAa,IAAI,IAAI,CAAC,eAAe,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YACtD,aAAa,GAAG,IAAI,CAAC,eAAe,CAAC,CAAC,CAAC,CAAC;QAC1C,CAAC;QAED,IAAI,aAAa,IAAI,aAAa,CAAC,EAAE,KAAK,IAAI,CAAC,QAAQ,EAAE,EAAE,CAAC;YAC1D,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,GAAG,EAAE;gBACnB,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,aAAc,CAAC,EAAE,CAAC,CAAC;gBACrC,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,aAAc,CAAC,EAAE,CAAC,CAAC;YAC7C,CAAC,CAAC,CAAC;QACL,CAAC;IACH,CAAC;IAEO,gBAAgB;QACtB,IAAI,IAAI,CAAC,aAAa,EAAE,CAAC;YACvB,MAAM,CAAC,mBAAmB,CAAC,QAAQ,EAAE,IAAI,CAAC,aAAa,CAAC,CAAC;YACzD,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC;QAC5B,CAAC;QACD,IAAI,IAAI,CAAC,KAAK,EAAE,CAAC;YACf,oBAAoB,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;YACjC,IAAI,CAAC,KAAK,GAAG,IAAI,CAAC;QACpB,CAAC;QACD,IAAI,CAAC,eAAe,GAAG,EAAE,CAAC;IAC5B,CAAC;IAED,eAAe,CAAC,KAAY,EAAE,EAAU;QACtC,KAAK,CAAC,cAAc,EAAE,CAAC;QAEvB,MAAM,OAAO,GAAG,QAAQ,CAAC,cAAc,CAAC,EAAE,CAAC,CAAC;QAC5C,IAAI,CAAC,OAAO;YAAE,OAAO;QAErB,MAAM,SAAS,GAAG,IAAI,CAAC,KAAK,CAAC,SAAS,IAAI,GAAG,CAAC;QAE9C,MAAM,GAAG,GAAG,OAAO,CAAC,qBAAqB,EAAE,CAAC,GAAG,GAAG,MAAM,CAAC,OAAO,GAAG,SAAS,CAAC;QAE7E,MAAM,CAAC,QAAQ,CAAC;YACd,GAAG;YACH,QAAQ,EAAE,QAAQ;SACnB,CAAC,CAAC;QAEH,0CAA0C;QAC1C,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;QAEtB,kCAAkC;QAClC,OAAO,CAAC,SAAS,CAAC,IAAI,EAAE,EAAE,EAAE,IAAI,EAAE,EAAE,CAAC,CAAC;IACxC,CAAC;+GAzMU,gBAAgB;mGAAhB,gBAAgB,sKAlCjB;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA+BT,igDAhCS,YAAY;;4FAmCX,gBAAgB;kBAtC5B,SAAS;+BACE,cAAc,cACZ,IAAI,WACP,CAAC,YAAY,CAAC,YACb;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA+BT;2EAIQ,KAAK;sBAAb,KAAK;gBAEI,aAAa;sBAAtB,MAAM","sourcesContent":["import { CommonModule } from '@angular/common';\nimport {\n  Component,\n  Input,\n  Output,\n  EventEmitter,\n  OnInit,\n  OnDestroy,\n  OnChanges,\n  SimpleChanges,\n  AfterViewInit,\n  NgZone,\n  signal,\n} from '@angular/core';\nimport { DocsTocMetadata, DocsTocItem } from './types';\n\n/**\n * val-docs-toc\n *\n * A table of contents component with scroll spy functionality.\n * Automatically highlights the current section based on scroll position.\n *\n * @example Manual items\n * ```html\n * <val-docs-toc\n *   [props]=\"{\n *     title: 'On this page',\n *     items: [\n *       { id: 'overview', label: 'Overview', level: 1 },\n *       { id: 'installation', label: 'Installation', level: 1 },\n *       { id: 'npm', label: 'Using npm', level: 2 },\n *       { id: 'yarn', label: 'Using yarn', level: 2 }\n *     ]\n *   }\"\n * ></val-docs-toc>\n * ```\n *\n * @example Auto-generated from headings\n * ```html\n * <val-docs-toc\n *   [props]=\"{\n *     autoGenerate: true,\n *     containerSelector: '.article-content',\n *     headingLevels: [2, 3]\n *   }\"\n * ></val-docs-toc>\n * ```\n */\n@Component({\n  selector: 'val-docs-toc',\n  standalone: true,\n  imports: [CommonModule],\n  template: `\n    <nav\n      class=\"docs-toc\"\n      [class]=\"props.cssClass\"\n      role=\"navigation\"\n      aria-label=\"Table of contents\"\n    >\n      @if (!props.hideTitle) {\n        <h4 class=\"docs-toc__title\">{{ props.title || 'Contents' }}</h4>\n      }\n\n      <ul class=\"docs-toc__list\">\n        @for (item of flatItems(); track item.id) {\n          <li\n            class=\"docs-toc__item\"\n            [class.docs-toc__item--level-1]=\"item.level === 1\"\n            [class.docs-toc__item--level-2]=\"item.level === 2\"\n            [class.docs-toc__item--level-3]=\"item.level === 3\"\n            [class.docs-toc__item--active]=\"activeId() === item.id\"\n          >\n            <a\n              class=\"docs-toc__link\"\n              [href]=\"'#' + item.id\"\n              (click)=\"scrollToSection($event, item.id)\"\n            >\n              {{ item.label }}\n            </a>\n          </li>\n        }\n      </ul>\n    </nav>\n  `,\n  styleUrls: ['./docs-toc.component.scss'],\n})\nexport class DocsTocComponent implements OnInit, OnDestroy, OnChanges, AfterViewInit {\n  @Input() props: DocsTocMetadata = {};\n\n  @Output() sectionChange = new EventEmitter<string>();\n\n  // Reactive state\n  protected activeId = signal<string>('');\n  protected flatItems = signal<DocsTocItem[]>([]);\n\n  private headingElements: HTMLElement[] = [];\n  private scrollHandler: (() => void) | null = null;\n  private rafId: number | null = null;\n\n  constructor(private ngZone: NgZone) {}\n\n  ngOnInit(): void {\n    this.updateFlatItems();\n  }\n\n  ngAfterViewInit(): void {\n    // Delay to ensure DOM is ready\n    setTimeout(() => {\n      if (this.props.autoGenerate) {\n        this.generateTocFromHeadings();\n      }\n      this.setupScrollSpy();\n    }, 100);\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes['props']) {\n      this.updateFlatItems();\n      if (this.props.autoGenerate) {\n        setTimeout(() => this.generateTocFromHeadings(), 100);\n      }\n    }\n  }\n\n  ngOnDestroy(): void {\n    this.destroyScrollSpy();\n  }\n\n  private updateFlatItems(): void {\n    if (this.props.items) {\n      this.flatItems.set(this.flattenItems(this.props.items));\n    }\n  }\n\n  private flattenItems(items: DocsTocItem[]): DocsTocItem[] {\n    const result: DocsTocItem[] = [];\n    const flatten = (itemList: DocsTocItem[]) => {\n      itemList.forEach(item => {\n        result.push(item);\n        if (item.children?.length) {\n          flatten(item.children);\n        }\n      });\n    };\n    flatten(items);\n    return result;\n  }\n\n  private generateTocFromHeadings(): void {\n    const selector = this.props.containerSelector || '.docs-layout__content';\n    const container = document.querySelector(selector);\n\n    if (!container) {\n      console.warn(`[docs-toc] Container not found: ${selector}`);\n      return;\n    }\n\n    const levels = this.props.headingLevels || [2, 3];\n    const headingSelector = levels.map(l => `h${l}`).join(', ');\n    const headings = container.querySelectorAll(headingSelector);\n\n    const items: DocsTocItem[] = [];\n    headings.forEach((heading: Element) => {\n      const el = heading as HTMLElement;\n      // Generate ID if not present\n      if (!el.id) {\n        el.id = this.generateId(el.textContent || '');\n      }\n\n      const level = parseInt(el.tagName[1], 10);\n      items.push({\n        id: el.id,\n        label: el.textContent?.trim() || '',\n        level: level - 1, // h2 = level 1, h3 = level 2\n      });\n    });\n\n    this.flatItems.set(items);\n    // Re-setup scroll spy with new items\n    this.setupScrollSpy();\n  }\n\n  private generateId(text: string): string {\n    return text\n      .toLowerCase()\n      .trim()\n      .replace(/[^\\w\\s-]/g, '')\n      .replace(/\\s+/g, '-')\n      .substring(0, 50);\n  }\n\n  private setupScrollSpy(): void {\n    this.destroyScrollSpy();\n\n    const items = this.flatItems();\n    if (items.length === 0) return;\n\n    // Collect heading elements\n    this.headingElements = items\n      .map(item => document.getElementById(item.id))\n      .filter((el): el is HTMLElement => el !== null);\n\n    if (this.headingElements.length === 0) return;\n\n    // Set initial active item\n    this.updateActiveHeading();\n\n    // Use scroll event with requestAnimationFrame for performance\n    this.ngZone.runOutsideAngular(() => {\n      this.scrollHandler = () => {\n        if (this.rafId) {\n          cancelAnimationFrame(this.rafId);\n        }\n        this.rafId = requestAnimationFrame(() => {\n          this.updateActiveHeading();\n        });\n      };\n\n      window.addEventListener('scroll', this.scrollHandler, { passive: true });\n    });\n  }\n\n  private updateActiveHeading(): void {\n    const offsetTop = this.props.offsetTop ?? 100;\n    const scrollY = window.scrollY;\n\n    // Find the heading that is currently at or above the offset point\n    let activeHeading: HTMLElement | null = null;\n\n    for (const heading of this.headingElements) {\n      const rect = heading.getBoundingClientRect();\n      const headingTop = rect.top + scrollY;\n\n      // If heading is at or above the trigger point (scrollY + offset)\n      if (headingTop <= scrollY + offsetTop + 10) {\n        activeHeading = heading;\n      } else {\n        // Headings are in order, so we can stop once we find one below\n        break;\n      }\n    }\n\n    // If no heading is above the trigger point, use the first one\n    if (!activeHeading && this.headingElements.length > 0) {\n      activeHeading = this.headingElements[0];\n    }\n\n    if (activeHeading && activeHeading.id !== this.activeId()) {\n      this.ngZone.run(() => {\n        this.activeId.set(activeHeading!.id);\n        this.sectionChange.emit(activeHeading!.id);\n      });\n    }\n  }\n\n  private destroyScrollSpy(): void {\n    if (this.scrollHandler) {\n      window.removeEventListener('scroll', this.scrollHandler);\n      this.scrollHandler = null;\n    }\n    if (this.rafId) {\n      cancelAnimationFrame(this.rafId);\n      this.rafId = null;\n    }\n    this.headingElements = [];\n  }\n\n  scrollToSection(event: Event, id: string): void {\n    event.preventDefault();\n\n    const element = document.getElementById(id);\n    if (!element) return;\n\n    const offsetTop = this.props.offsetTop ?? 100;\n\n    const top = element.getBoundingClientRect().top + window.scrollY - offsetTop;\n\n    window.scrollTo({\n      top,\n      behavior: 'smooth',\n    });\n\n    // Update active immediately for better UX\n    this.activeId.set(id);\n\n    // Update URL hash without jumping\n    history.pushState(null, '', `#${id}`);\n  }\n}\n"]}
318
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"docs-toc.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/organisms/docs-toc/docs-toc.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport {\n  Component,\n  Input,\n  Output,\n  EventEmitter,\n  OnInit,\n  OnDestroy,\n  OnChanges,\n  SimpleChanges,\n  AfterViewInit,\n  NgZone,\n  signal,\n} from '@angular/core';\nimport { DocsTocMetadata, DocsTocItem } from './types';\n\n/**\n * val-docs-toc\n *\n * A table of contents component with scroll spy functionality.\n * Automatically highlights the current section based on scroll position.\n *\n * @example Manual items\n * ```html\n * <val-docs-toc\n *   [props]=\"{\n *     title: 'On this page',\n *     items: [\n *       { id: 'overview', label: 'Overview', level: 1 },\n *       { id: 'installation', label: 'Installation', level: 1 },\n *       { id: 'npm', label: 'Using npm', level: 2 },\n *       { id: 'yarn', label: 'Using yarn', level: 2 }\n *     ]\n *   }\"\n * ></val-docs-toc>\n * ```\n *\n * @example Auto-generated from headings\n * ```html\n * <val-docs-toc\n *   [props]=\"{\n *     autoGenerate: true,\n *     containerSelector: '.article-content',\n *     headingLevels: [2, 3]\n *   }\"\n * ></val-docs-toc>\n * ```\n */\n@Component({\n  selector: 'val-docs-toc',\n  standalone: true,\n  imports: [CommonModule],\n  template: `\n    <nav\n      class=\"docs-toc\"\n      [class]=\"props.cssClass\"\n      role=\"navigation\"\n      aria-label=\"Table of contents\"\n    >\n      @if (!props.hideTitle) {\n        <h4 class=\"docs-toc__title\">{{ props.title || 'Contents' }}</h4>\n      }\n\n      <ul class=\"docs-toc__list\">\n        @for (item of flatItems(); track item.id) {\n          <li\n            class=\"docs-toc__item\"\n            [class.docs-toc__item--level-1]=\"item.level === 1\"\n            [class.docs-toc__item--level-2]=\"item.level === 2\"\n            [class.docs-toc__item--level-3]=\"item.level === 3\"\n            [class.docs-toc__item--active]=\"activeId() === item.id\"\n          >\n            <a\n              class=\"docs-toc__link\"\n              [href]=\"'#' + item.id\"\n              (click)=\"scrollToSection($event, item.id)\"\n            >\n              {{ item.label }}\n            </a>\n          </li>\n        }\n      </ul>\n    </nav>\n  `,\n  styleUrls: ['./docs-toc.component.scss'],\n})\nexport class DocsTocComponent implements OnInit, OnDestroy, OnChanges, AfterViewInit {\n  @Input() props: DocsTocMetadata = {};\n\n  @Output() sectionChange = new EventEmitter<string>();\n\n  // Reactive state\n  protected activeId = signal<string>('');\n  protected flatItems = signal<DocsTocItem[]>([]);\n\n  private headingElements: HTMLElement[] = [];\n  private scrollHandler: (() => void) | null = null;\n  private rafId: number | null = null;\n\n  constructor(private ngZone: NgZone) {}\n\n  ngOnInit(): void {\n    this.updateFlatItems();\n  }\n\n  ngAfterViewInit(): void {\n    // Delay to ensure DOM is ready\n    setTimeout(() => {\n      if (this.props.autoGenerate) {\n        this.generateTocFromHeadings();\n      }\n      this.setupScrollSpy();\n    }, 100);\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes['props']) {\n      this.updateFlatItems();\n      if (this.props.autoGenerate) {\n        setTimeout(() => this.generateTocFromHeadings(), 100);\n      }\n    }\n  }\n\n  ngOnDestroy(): void {\n    this.destroyScrollSpy();\n  }\n\n  private updateFlatItems(): void {\n    if (this.props.items) {\n      this.flatItems.set(this.flattenItems(this.props.items));\n    }\n  }\n\n  private flattenItems(items: DocsTocItem[]): DocsTocItem[] {\n    const result: DocsTocItem[] = [];\n    const flatten = (itemList: DocsTocItem[]) => {\n      itemList.forEach(item => {\n        result.push(item);\n        if (item.children?.length) {\n          flatten(item.children);\n        }\n      });\n    };\n    flatten(items);\n    return result;\n  }\n\n  private generateTocFromHeadings(): void {\n    const selector = this.props.containerSelector || '.docs-layout__content';\n    const container = document.querySelector(selector);\n\n    if (!container) {\n      console.warn(`[docs-toc] Container not found: ${selector}`);\n      return;\n    }\n\n    const levels = this.props.headingLevels || [2, 3];\n    const headingSelector = levels.map(l => `h${l}`).join(', ');\n    const headings = container.querySelectorAll(headingSelector);\n\n    const items: DocsTocItem[] = [];\n    headings.forEach((heading: Element) => {\n      const el = heading as HTMLElement;\n      // Generate ID if not present\n      if (!el.id) {\n        el.id = this.generateId(el.textContent || '');\n      }\n\n      const level = parseInt(el.tagName[1], 10);\n      items.push({\n        id: el.id,\n        label: el.textContent?.trim() || '',\n        level: level - 1, // h2 = level 1, h3 = level 2\n      });\n    });\n\n    this.flatItems.set(items);\n    // Re-setup scroll spy with new items\n    this.setupScrollSpy();\n  }\n\n  private generateId(text: string): string {\n    return text\n      .toLowerCase()\n      .trim()\n      .replace(/[^\\w\\s-]/g, '')\n      .replace(/\\s+/g, '-')\n      .substring(0, 50);\n  }\n\n  private scrollContainer: HTMLElement | Window | null = null;\n\n  private setupScrollSpy(): void {\n    this.destroyScrollSpy();\n\n    const items = this.flatItems();\n    if (items.length === 0) return;\n\n    // Collect heading elements\n    this.headingElements = items\n      .map(item => document.getElementById(item.id))\n      .filter((el): el is HTMLElement => el !== null);\n\n    if (this.headingElements.length === 0) {\n      console.warn('[docs-toc] No heading elements found for IDs:', items.map(i => i.id));\n      return;\n    }\n\n    // Find the scroll container - check for ion-content first, then use window\n    this.scrollContainer = this.findScrollContainer();\n\n    // Set initial active item\n    this.updateActiveHeading();\n\n    // Use scroll event with requestAnimationFrame for performance\n    this.ngZone.runOutsideAngular(() => {\n      this.scrollHandler = () => {\n        if (this.rafId) {\n          cancelAnimationFrame(this.rafId);\n        }\n        this.rafId = requestAnimationFrame(() => {\n          this.updateActiveHeading();\n        });\n      };\n\n      // Listen on both window and the scroll container for maximum compatibility\n      window.addEventListener('scroll', this.scrollHandler, { passive: true });\n      document.addEventListener('scroll', this.scrollHandler, { passive: true, capture: true });\n\n      // Also listen on ion-content if present\n      const ionContent = document.querySelector('ion-content');\n      if (ionContent) {\n        ionContent.addEventListener('ionScroll', this.scrollHandler as EventListener, { passive: true });\n      }\n    });\n  }\n\n  private findScrollContainer(): HTMLElement | Window {\n    // Check for ion-content's scroll element\n    const ionContent = document.querySelector('ion-content');\n    if (ionContent) {\n      // ion-content has a shadow DOM with the actual scrollable element\n      const scrollEl = (ionContent as any).shadowRoot?.querySelector('.inner-scroll');\n      if (scrollEl) return scrollEl;\n    }\n    return window;\n  }\n\n  private updateActiveHeading(): void {\n    const offsetTop = this.props.offsetTop ?? 100;\n\n    // Find the heading closest to or above the trigger point (offset from top of viewport)\n    // Using getBoundingClientRect().top which is relative to viewport - works regardless of scroll container\n    let activeHeading: HTMLElement | null = null;\n\n    for (const heading of this.headingElements) {\n      const rect = heading.getBoundingClientRect();\n\n      // If heading's top is at or above the trigger point (offset pixels from viewport top)\n      if (rect.top <= offsetTop + 10) {\n        activeHeading = heading;\n      } else {\n        // Since headings are in DOM order, stop once we find one below the trigger\n        break;\n      }\n    }\n\n    // If no heading is above the trigger point, use the first one\n    if (!activeHeading && this.headingElements.length > 0) {\n      activeHeading = this.headingElements[0];\n    }\n\n    if (activeHeading && activeHeading.id !== this.activeId()) {\n      this.ngZone.run(() => {\n        this.activeId.set(activeHeading!.id);\n        this.sectionChange.emit(activeHeading!.id);\n      });\n    }\n  }\n\n  private destroyScrollSpy(): void {\n    if (this.scrollHandler) {\n      window.removeEventListener('scroll', this.scrollHandler);\n      document.removeEventListener('scroll', this.scrollHandler, { capture: true } as EventListenerOptions);\n\n      const ionContent = document.querySelector('ion-content');\n      if (ionContent) {\n        ionContent.removeEventListener('ionScroll', this.scrollHandler as EventListener);\n      }\n\n      this.scrollHandler = null;\n    }\n    if (this.rafId) {\n      cancelAnimationFrame(this.rafId);\n      this.rafId = null;\n    }\n    this.headingElements = [];\n    this.scrollContainer = null;\n  }\n\n  scrollToSection(event: Event, id: string): void {\n    event.preventDefault();\n\n    const element = document.getElementById(id);\n    if (!element) return;\n\n    const offsetTop = this.props.offsetTop ?? 100;\n\n    // Use scrollIntoView with offset calculation for better cross-container support\n    const elementRect = element.getBoundingClientRect();\n    const currentScrollY = window.scrollY || document.documentElement.scrollTop;\n    const targetY = elementRect.top + currentScrollY - offsetTop;\n\n    // Try ion-content scrollToPoint first (for Ionic apps)\n    const ionContent = document.querySelector('ion-content');\n    if (ionContent && typeof (ionContent as any).scrollToPoint === 'function') {\n      (ionContent as any).scrollToPoint(0, targetY, 300);\n    } else {\n      // Fallback to window scroll\n      window.scrollTo({\n        top: targetY,\n        behavior: 'smooth',\n      });\n    }\n\n    // Update active immediately for better UX\n    this.activeId.set(id);\n\n    // Update URL hash without jumping\n    history.pushState(null, '', `#${id}`);\n  }\n}\n"]}
@@ -14782,8 +14782,20 @@ class ParticipantCardComponent {
14782
14782
  const status = this.props.participant.paymentStatus;
14783
14783
  if (!status)
14784
14784
  return '';
14785
- const labels = { ...DEFAULT_PAYMENT_STATUS_LABELS, ...this.props.paymentStatusLabels };
14786
- return labels[status] || status;
14785
+ // If custom labels are provided, use them first
14786
+ if (this.props.paymentStatusLabels?.[status]) {
14787
+ return this.props.paymentStatusLabels[status];
14788
+ }
14789
+ // Use i18n for default labels
14790
+ const i18nKeys = {
14791
+ pending: 'paymentPending',
14792
+ processing: 'paymentProcessing',
14793
+ paid: 'paymentPaid',
14794
+ refunded: 'paymentRefunded',
14795
+ failed: 'paymentFailed',
14796
+ };
14797
+ const key = i18nKeys[status];
14798
+ return key ? this.i18n.t(key) : status;
14787
14799
  }
14788
14800
  getPaymentStatusColor() {
14789
14801
  const status = this.props.participant.paymentStatus;
@@ -32802,6 +32814,7 @@ class DocsTocComponent {
32802
32814
  this.headingElements = [];
32803
32815
  this.scrollHandler = null;
32804
32816
  this.rafId = null;
32817
+ this.scrollContainer = null;
32805
32818
  }
32806
32819
  ngOnInit() {
32807
32820
  this.updateFlatItems();
@@ -32889,8 +32902,12 @@ class DocsTocComponent {
32889
32902
  this.headingElements = items
32890
32903
  .map(item => document.getElementById(item.id))
32891
32904
  .filter((el) => el !== null);
32892
- if (this.headingElements.length === 0)
32905
+ if (this.headingElements.length === 0) {
32906
+ console.warn('[docs-toc] No heading elements found for IDs:', items.map(i => i.id));
32893
32907
  return;
32908
+ }
32909
+ // Find the scroll container - check for ion-content first, then use window
32910
+ this.scrollContainer = this.findScrollContainer();
32894
32911
  // Set initial active item
32895
32912
  this.updateActiveHeading();
32896
32913
  // Use scroll event with requestAnimationFrame for performance
@@ -32903,23 +32920,40 @@ class DocsTocComponent {
32903
32920
  this.updateActiveHeading();
32904
32921
  });
32905
32922
  };
32923
+ // Listen on both window and the scroll container for maximum compatibility
32906
32924
  window.addEventListener('scroll', this.scrollHandler, { passive: true });
32925
+ document.addEventListener('scroll', this.scrollHandler, { passive: true, capture: true });
32926
+ // Also listen on ion-content if present
32927
+ const ionContent = document.querySelector('ion-content');
32928
+ if (ionContent) {
32929
+ ionContent.addEventListener('ionScroll', this.scrollHandler, { passive: true });
32930
+ }
32907
32931
  });
32908
32932
  }
32933
+ findScrollContainer() {
32934
+ // Check for ion-content's scroll element
32935
+ const ionContent = document.querySelector('ion-content');
32936
+ if (ionContent) {
32937
+ // ion-content has a shadow DOM with the actual scrollable element
32938
+ const scrollEl = ionContent.shadowRoot?.querySelector('.inner-scroll');
32939
+ if (scrollEl)
32940
+ return scrollEl;
32941
+ }
32942
+ return window;
32943
+ }
32909
32944
  updateActiveHeading() {
32910
32945
  const offsetTop = this.props.offsetTop ?? 100;
32911
- const scrollY = window.scrollY;
32912
- // Find the heading that is currently at or above the offset point
32946
+ // Find the heading closest to or above the trigger point (offset from top of viewport)
32947
+ // Using getBoundingClientRect().top which is relative to viewport - works regardless of scroll container
32913
32948
  let activeHeading = null;
32914
32949
  for (const heading of this.headingElements) {
32915
32950
  const rect = heading.getBoundingClientRect();
32916
- const headingTop = rect.top + scrollY;
32917
- // If heading is at or above the trigger point (scrollY + offset)
32918
- if (headingTop <= scrollY + offsetTop + 10) {
32951
+ // If heading's top is at or above the trigger point (offset pixels from viewport top)
32952
+ if (rect.top <= offsetTop + 10) {
32919
32953
  activeHeading = heading;
32920
32954
  }
32921
32955
  else {
32922
- // Headings are in order, so we can stop once we find one below
32956
+ // Since headings are in DOM order, stop once we find one below the trigger
32923
32957
  break;
32924
32958
  }
32925
32959
  }
@@ -32937,6 +32971,11 @@ class DocsTocComponent {
32937
32971
  destroyScrollSpy() {
32938
32972
  if (this.scrollHandler) {
32939
32973
  window.removeEventListener('scroll', this.scrollHandler);
32974
+ document.removeEventListener('scroll', this.scrollHandler, { capture: true });
32975
+ const ionContent = document.querySelector('ion-content');
32976
+ if (ionContent) {
32977
+ ionContent.removeEventListener('ionScroll', this.scrollHandler);
32978
+ }
32940
32979
  this.scrollHandler = null;
32941
32980
  }
32942
32981
  if (this.rafId) {
@@ -32944,6 +32983,7 @@ class DocsTocComponent {
32944
32983
  this.rafId = null;
32945
32984
  }
32946
32985
  this.headingElements = [];
32986
+ this.scrollContainer = null;
32947
32987
  }
32948
32988
  scrollToSection(event, id) {
32949
32989
  event.preventDefault();
@@ -32951,11 +32991,22 @@ class DocsTocComponent {
32951
32991
  if (!element)
32952
32992
  return;
32953
32993
  const offsetTop = this.props.offsetTop ?? 100;
32954
- const top = element.getBoundingClientRect().top + window.scrollY - offsetTop;
32955
- window.scrollTo({
32956
- top,
32957
- behavior: 'smooth',
32958
- });
32994
+ // Use scrollIntoView with offset calculation for better cross-container support
32995
+ const elementRect = element.getBoundingClientRect();
32996
+ const currentScrollY = window.scrollY || document.documentElement.scrollTop;
32997
+ const targetY = elementRect.top + currentScrollY - offsetTop;
32998
+ // Try ion-content scrollToPoint first (for Ionic apps)
32999
+ const ionContent = document.querySelector('ion-content');
33000
+ if (ionContent && typeof ionContent.scrollToPoint === 'function') {
33001
+ ionContent.scrollToPoint(0, targetY, 300);
33002
+ }
33003
+ else {
33004
+ // Fallback to window scroll
33005
+ window.scrollTo({
33006
+ top: targetY,
33007
+ behavior: 'smooth',
33008
+ });
33009
+ }
32959
33010
  // Update active immediately for better UX
32960
33011
  this.activeId.set(id);
32961
33012
  // Update URL hash without jumping