133 lines
3.1 KiB
TypeScript
133 lines
3.1 KiB
TypeScript
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: `
|
|
<div
|
|
#cardElement
|
|
class="w-24 h-36 rounded-lg p-2 relative flex flex-col justify-between shadow-lg card-element"
|
|
[class]="hidden ? 'bg-red-800' : 'bg-white'"
|
|
>
|
|
@if (!hidden) {
|
|
<span class="text-xl font-bold" [class]="isRedSuit ? 'text-accent-red' : 'text-black'">{{
|
|
getDisplayRank(rank)
|
|
}}</span>
|
|
}
|
|
@if (!hidden) {
|
|
<span
|
|
class="absolute top-1/2 left-1/2 -translate-x-1/2 -translate-y-1/2 text-3xl"
|
|
[class]="isRedSuit ? 'text-accent-red' : 'text-black'"
|
|
>{{ getSuitSymbol(suit) }}</span
|
|
>
|
|
}
|
|
@if (!hidden) {
|
|
<span
|
|
class="text-xl font-bold self-end rotate-180"
|
|
[class]="isRedSuit ? 'text-accent-red' : 'text-black'"
|
|
>{{ getDisplayRank(rank) }}</span
|
|
>
|
|
}
|
|
</div>
|
|
`,
|
|
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<string, string> = {
|
|
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;
|
|
}
|
|
}
|