import { ChangeDetectionStrategy, Component, Input, AfterViewInit, ElementRef, OnChanges, SimpleChanges, } from '@angular/core'; import { CommonModule } from '@angular/common'; import { gsap } from 'gsap'; import { Suit, suitSymbols } from '@blackjack/models/blackjack.model'; @Component({ selector: 'app-playing-card', standalone: true, imports: [CommonModule], template: `
@if (!hidden) { {{ getDisplayRank(rank) }} } @if (!hidden) { {{ getSuitSymbol(suit) }} } @if (!hidden) { {{ getDisplayRank(rank) }} }
`, styles: [ ` .card-element { transform-style: preserve-3d; backface-visibility: hidden; } `, ], changeDetection: ChangeDetectionStrategy.OnPush, }) export class PlayingCardComponent implements AfterViewInit, OnChanges { @Input({ required: true }) rank!: string; @Input({ required: true }) suit!: Suit; @Input({ required: true }) hidden!: boolean; @Input() isNew = false; constructor(private elementRef: ElementRef) {} get isRedSuit(): boolean { return this.suit === 'HEARTS' || this.suit === 'DIAMONDS'; } ngAfterViewInit(): void { if (this.isNew) { this.animateNewCard(); } } ngOnChanges(changes: SimpleChanges): void { if (changes['hidden'] && !changes['hidden'].firstChange) { this.animateCardFlip(); } } private animateNewCard(): void { const cardElement = this.elementRef.nativeElement.querySelector('.card-element'); gsap.fromTo( cardElement, { y: -100, opacity: 0, rotation: -10, scale: 0.7, }, { y: 0, opacity: 1, rotation: 0, scale: 1, duration: 0.5, ease: 'power2.out', } ); } private animateCardFlip(): void { const cardElement = this.elementRef.nativeElement.querySelector('.card-element'); gsap.to(cardElement, { rotationY: 180, duration: 0.3, onComplete: () => { gsap.set(cardElement, { rotationY: 0 }); }, }); } protected getSuitSymbol(suit: Suit): string { return suitSymbols[suit]; } protected getDisplayRank(rank: string): string { const rankMap: Record = { TWO: '2', THREE: '3', FOUR: '4', FIVE: '5', SIX: '6', SEVEN: '7', EIGHT: '8', NINE: '9', TEN: '10', JACK: 'J', QUEEN: 'Q', KING: 'K', ACE: 'A', }; return rankMap[rank] || rank; } }