valtech-components 2.0.538 → 2.0.539
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 {
|
|
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
|
-
|
|
148
|
-
|
|
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"]}
|
|
@@ -14782,8 +14782,20 @@ class ParticipantCardComponent {
|
|
|
14782
14782
|
const status = this.props.participant.paymentStatus;
|
|
14783
14783
|
if (!status)
|
|
14784
14784
|
return '';
|
|
14785
|
-
|
|
14786
|
-
|
|
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;
|