From eb153f4459e53848875a1b088e269cfa413c4eaa Mon Sep 17 00:00:00 2001 From: Jan-Marlon Leibl Date: Wed, 26 Mar 2025 13:26:38 +0100 Subject: [PATCH] feat(game): add blackjack game component and routing --- frontend/src/app/app.routes.ts | 5 ++ .../game/blackjack/blackjack.component.html | 23 +++++++ .../game/blackjack/blackjack.component.ts | 63 +++++++++++++++++++ .../dealer-hand/dealer-hand.component.ts | 29 +++++++++ .../game-controls/game-controls.component.ts | 36 +++++++++++ .../game-info/game-info.component.ts | 34 ++++++++++ .../player-hand/player-hand.component.ts | 31 +++++++++ .../playing-card/playing-card.component.ts | 30 +++++++++ .../game/blackjack/models/card.model.ts | 5 ++ .../src/app/feature/home/home.component.html | 8 ++- .../src/app/feature/home/home.component.ts | 10 +++ frontend/src/app/model/Game.ts | 1 + 12 files changed, 273 insertions(+), 2 deletions(-) create mode 100644 frontend/src/app/feature/game/blackjack/blackjack.component.html create mode 100644 frontend/src/app/feature/game/blackjack/blackjack.component.ts create mode 100644 frontend/src/app/feature/game/blackjack/components/dealer-hand/dealer-hand.component.ts create mode 100644 frontend/src/app/feature/game/blackjack/components/game-controls/game-controls.component.ts create mode 100644 frontend/src/app/feature/game/blackjack/components/game-info/game-info.component.ts create mode 100644 frontend/src/app/feature/game/blackjack/components/player-hand/player-hand.component.ts create mode 100644 frontend/src/app/feature/game/blackjack/components/playing-card/playing-card.component.ts create mode 100644 frontend/src/app/feature/game/blackjack/models/card.model.ts diff --git a/frontend/src/app/app.routes.ts b/frontend/src/app/app.routes.ts index 6fbef95..e783513 100644 --- a/frontend/src/app/app.routes.ts +++ b/frontend/src/app/app.routes.ts @@ -16,4 +16,9 @@ export const routes: Routes = [ loadComponent: () => import('./feature/home/home.component'), canActivate: [authGuard], }, + { + path: 'game/blackjack', + loadComponent: () => import('./feature/game/blackjack/blackjack.component'), + canActivate: [authGuard], + }, ]; diff --git a/frontend/src/app/feature/game/blackjack/blackjack.component.html b/frontend/src/app/feature/game/blackjack/blackjack.component.html new file mode 100644 index 0000000..dd62189 --- /dev/null +++ b/frontend/src/app/feature/game/blackjack/blackjack.component.html @@ -0,0 +1,23 @@ + + +
+
+
+ + + +
+ +
+ +
+
+
diff --git a/frontend/src/app/feature/game/blackjack/blackjack.component.ts b/frontend/src/app/feature/game/blackjack/blackjack.component.ts new file mode 100644 index 0000000..4e66587 --- /dev/null +++ b/frontend/src/app/feature/game/blackjack/blackjack.component.ts @@ -0,0 +1,63 @@ +import { ChangeDetectionStrategy, Component, inject, signal } from '@angular/core'; +import { CommonModule } from '@angular/common'; +import { NavbarComponent } from '../../../shared/components/navbar/navbar.component'; +import { Router } from '@angular/router'; +import { UserService } from '../../../service/user.service'; +import { PlayingCardComponent } from './components/playing-card/playing-card.component'; +import { DealerHandComponent } from './components/dealer-hand/dealer-hand.component'; +import { PlayerHandComponent } from './components/player-hand/player-hand.component'; +import { GameControlsComponent } from './components/game-controls/game-controls.component'; +import { GameInfoComponent } from './components/game-info/game-info.component'; +import { Card } from './models/card.model'; + +@Component({ + selector: 'app-blackjack', + standalone: true, + imports: [ + CommonModule, + NavbarComponent, + PlayingCardComponent, + DealerHandComponent, + PlayerHandComponent, + GameControlsComponent, + GameInfoComponent, + ], + templateUrl: './blackjack.component.html', + changeDetection: ChangeDetectionStrategy.OnPush, +}) +export default class BlackjackComponent { + private router = inject(Router); + private userService = inject(UserService); + + dealerCards: Card[] = [ + { value: '2', suit: '♥', hidden: false }, + { value: '3', suit: '♦', hidden: false }, + { value: 'B', suit: '□', hidden: true }, + ]; + + playerCards: Card[] = []; + currentBet = 0; + balance = signal(0); + + constructor() { + this.userService.getCurrentUser().subscribe((user) => { + this.balance.set(user?.balance ?? 0); + }); + } + + onHit(): void { + // Implementation for hit action + } + + onStand(): void { + // Implementation for stand action + } + + leaveGame(): void { + this.router.navigate(['/home']); + } + + onNewGame(): void { + // Implementation for new game + } +} diff --git a/frontend/src/app/feature/game/blackjack/components/dealer-hand/dealer-hand.component.ts b/frontend/src/app/feature/game/blackjack/components/dealer-hand/dealer-hand.component.ts new file mode 100644 index 0000000..11f9e90 --- /dev/null +++ b/frontend/src/app/feature/game/blackjack/components/dealer-hand/dealer-hand.component.ts @@ -0,0 +1,29 @@ +import { ChangeDetectionStrategy, Component, Input } from '@angular/core'; +import { CommonModule } from '@angular/common'; +import { PlayingCardComponent } from '../playing-card/playing-card.component'; +import { Card } from '../../models/card.model'; + +@Component({ + selector: 'app-dealer-hand', + standalone: true, + imports: [CommonModule, PlayingCardComponent], + template: ` +
+

Croupier's Karten

+
+
+ +
+
+
+ `, + changeDetection: ChangeDetectionStrategy.OnPush, +}) +export class DealerHandComponent { + @Input() cards: Card[] = []; +} diff --git a/frontend/src/app/feature/game/blackjack/components/game-controls/game-controls.component.ts b/frontend/src/app/feature/game/blackjack/components/game-controls/game-controls.component.ts new file mode 100644 index 0000000..92aac24 --- /dev/null +++ b/frontend/src/app/feature/game/blackjack/components/game-controls/game-controls.component.ts @@ -0,0 +1,36 @@ +import { ChangeDetectionStrategy, Component, EventEmitter, Output } from '@angular/core'; +import { CommonModule } from '@angular/common'; + +@Component({ + selector: 'app-game-controls', + standalone: true, + imports: [CommonModule], + template: ` +
+ + + +
+ `, + changeDetection: ChangeDetectionStrategy.OnPush, +}) +export class GameControlsComponent { + @Output() onHit = new EventEmitter(); + @Output() onStand = new EventEmitter(); + @Output() onLeave = new EventEmitter(); +} diff --git a/frontend/src/app/feature/game/blackjack/components/game-info/game-info.component.ts b/frontend/src/app/feature/game/blackjack/components/game-info/game-info.component.ts new file mode 100644 index 0000000..ab3f612 --- /dev/null +++ b/frontend/src/app/feature/game/blackjack/components/game-info/game-info.component.ts @@ -0,0 +1,34 @@ +import { ChangeDetectionStrategy, Component, EventEmitter, Input, Output } from '@angular/core'; +import { CommonModule, CurrencyPipe } from '@angular/common'; + +@Component({ + selector: 'app-game-info', + standalone: true, + imports: [CommonModule, CurrencyPipe], + template: ` +
+

Spiel Informationen

+
+
+ Guthaben: + {{ balance | currency: 'EUR' }} +
+
+ Aktuelle Wette: + + {{ currentBet | currency: 'EUR' }} + +
+ +
+
+ `, + changeDetection: ChangeDetectionStrategy.OnPush, +}) +export class GameInfoComponent { + @Input() balance: number = 0; + @Input() currentBet: number = 0; + @Output() onNewGameClick = new EventEmitter(); +} diff --git a/frontend/src/app/feature/game/blackjack/components/player-hand/player-hand.component.ts b/frontend/src/app/feature/game/blackjack/components/player-hand/player-hand.component.ts new file mode 100644 index 0000000..c157968 --- /dev/null +++ b/frontend/src/app/feature/game/blackjack/components/player-hand/player-hand.component.ts @@ -0,0 +1,31 @@ +import { ChangeDetectionStrategy, Component, Input } from '@angular/core'; +import { CommonModule } from '@angular/common'; +import { PlayingCardComponent } from '../playing-card/playing-card.component'; +import { Card } from '../../models/card.model'; + +@Component({ + selector: 'app-player-hand', + standalone: true, + imports: [CommonModule, PlayingCardComponent], + template: ` +
+

Deine Karten

+
+
+ +
+
+
+ `, + changeDetection: ChangeDetectionStrategy.OnPush, +}) +export class PlayerHandComponent { + @Input() cards: Card[] = []; +} diff --git a/frontend/src/app/feature/game/blackjack/components/playing-card/playing-card.component.ts b/frontend/src/app/feature/game/blackjack/components/playing-card/playing-card.component.ts new file mode 100644 index 0000000..dc93df8 --- /dev/null +++ b/frontend/src/app/feature/game/blackjack/components/playing-card/playing-card.component.ts @@ -0,0 +1,30 @@ +import { ChangeDetectionStrategy, Component, Input } from '@angular/core'; +import { CommonModule } from '@angular/common'; + +@Component({ + selector: 'app-playing-card', + standalone: true, + imports: [CommonModule], + template: ` +
+ {{ value }} + {{ suit }} + {{ + value + }} +
+ `, + changeDetection: ChangeDetectionStrategy.OnPush, +}) +export class PlayingCardComponent { + @Input({ required: true }) value!: string; + @Input({ required: true }) suit!: string; + @Input({ required: true }) hidden!: boolean; +} diff --git a/frontend/src/app/feature/game/blackjack/models/card.model.ts b/frontend/src/app/feature/game/blackjack/models/card.model.ts new file mode 100644 index 0000000..79b5ad1 --- /dev/null +++ b/frontend/src/app/feature/game/blackjack/models/card.model.ts @@ -0,0 +1,5 @@ +export interface Card { + value: string; + suit: string; + hidden: boolean; +} diff --git a/frontend/src/app/feature/home/home.component.html b/frontend/src/app/feature/home/home.component.html index 0d4708a..b1bfd9b 100644 --- a/frontend/src/app/feature/home/home.component.html +++ b/frontend/src/app/feature/home/home.component.html @@ -34,7 +34,9 @@ class="absolute bottom-4 left-4 right-4 transform translate-y-4 group-hover:translate-y-0 transition-transform duration-300" >

{{ game.name }}

- + @@ -59,7 +61,9 @@ class="absolute bottom-4 left-4 right-4 transform translate-y-4 group-hover:translate-y-0 transition-transform duration-300" >

{{ game.name }}

- + diff --git a/frontend/src/app/feature/home/home.component.ts b/frontend/src/app/feature/home/home.component.ts index 5baf2a0..b3237ee 100644 --- a/frontend/src/app/feature/home/home.component.ts +++ b/frontend/src/app/feature/home/home.component.ts @@ -38,31 +38,37 @@ export default class HomeComponent implements OnInit { id: '1', name: 'Poker', image: '/poker.webp', + route: '/game/poker', }, { id: '2', name: 'Blackjack', image: '/blackjack.webp', + route: '/game/blackjack', }, { id: '3', name: 'Slots', image: '/slots.webp', + route: '/game/slots', }, { id: '4', name: 'Plinko', image: '/plinko.webp', + route: '/game/plinko', }, { id: '5', name: 'Liars Dice', image: '/liars-dice.webp', + route: '/game/liars-dice', }, { id: '6', name: 'Lootboxen', image: '/lootbox.webp', + route: '/game/lootbox', }, ]; @@ -83,4 +89,8 @@ export default class HomeComponent implements OnInit { closeDepositConfirmationModal() { this.isDepositSuccessful = false; } + + navigateToGame(route: string) { + this.router.navigate([route]); + } } diff --git a/frontend/src/app/model/Game.ts b/frontend/src/app/model/Game.ts index a3f3e74..9adbf7a 100644 --- a/frontend/src/app/model/Game.ts +++ b/frontend/src/app/model/Game.ts @@ -2,4 +2,5 @@ export interface Game { id: string; name: string; image: string; + route: string; }